From 39198164cd9f78f9d0b1a44c7ba2e94bb66df017 Mon Sep 17 00:00:00 2001 From: Alec Murphy Date: Tue, 25 Mar 2025 07:32:23 -0400 Subject: [PATCH] Meta: Add files to repository --- .clang-format | 14 + .gitattributes | 2 + .gitignore | 3 + .vscode/settings.json | 10 + .vscode/tasks.json | 1 + Applets/Calendar.applet/Run.HC | 5 + Applets/ClipManager.applet/Run.HC | 32 + Applets/NetworkStatus.applet/Run.HC | 101 + .../window_icon_16x16.png | Bin 0 -> 613 bytes Applets/VolumeSlider.applet/Run.HC | 71 + .../Accessories/Calculator.app/Run.HC | 100 + .../Calculator.app/window_icon_16x16.png | Bin 0 -> 619 bytes Applications/Accessories/Icon.png | Bin 0 -> 850 bytes Applications/OS/Icon.png | Bin 0 -> 1010 bytes Applications/OS/MenuBar.app/Run.HC | 126 + Applications/OS/MenuBar.app/system_menu.png | Bin 0 -> 869 bytes Applications/OS/ShutDown.app/Icon.png | Bin 0 -> 875 bytes Applications/OS/ShutDown.app/Run.HC | 25 + Applications/OS/TaskSwitcher.app/Run.HC | 198 + Applications/OS/TempleOS.app/Run.HC | 127 + .../OS/TempleOS.app/window_icon_16x16.png | Bin 0 -> 268 bytes Applications/OS/Terminal.app/Icon.png | Bin 0 -> 689 bytes Applications/OS/Terminal.app/Run.HC | 205 + Applications/OS/Wallpaper.app/Run.HC | 29 + Applications/TestApplication.app/Run.HC | 96 + Include/Gui.HC | 53 + Media/Sounds/Beep.wav | Bin 0 -> 15404 bytes Media/Themes/Umami/BitmapFont/menu.png | Bin 0 -> 726 bytes Media/Themes/Umami/BitmapFont/monospace.png | Bin 0 -> 1371 bytes Media/Themes/Umami/BitmapFont/sans.png | Bin 0 -> 606 bytes Media/Themes/Umami/Icon/actions/add.png | Bin 0 -> 506 bytes .../Umami/Icon/actions/address-book-new.png | Bin 0 -> 1083 bytes .../Umami/Icon/actions/appointment-new.png | Bin 0 -> 1477 bytes .../Themes/Umami/Icon/actions/appointment.png | Bin 0 -> 1477 bytes Media/Themes/Umami/Icon/actions/back.png | Bin 0 -> 1135 bytes .../Umami/Icon/actions/bookmark-new.png | Bin 0 -> 1101 bytes .../Umami/Icon/actions/bookmark_add.png | Bin 0 -> 1101 bytes .../Umami/Icon/actions/bookmarks_list_add.png | Bin 0 -> 1101 bytes Media/Themes/Umami/Icon/actions/bottom.png | Bin 0 -> 1111 bytes .../Themes/Umami/Icon/actions/centrejust.png | Bin 0 -> 694 bytes .../Themes/Umami/Icon/actions/contact-new.png | Bin 0 -> 1032 bytes .../Umami/Icon/actions/document-new.png | Bin 0 -> 851 bytes .../Umami/Icon/actions/document-open.png | Bin 0 -> 1093 bytes .../Icon/actions/document-print-preview.png | Bin 0 -> 1291 bytes .../Umami/Icon/actions/document-print.png | Bin 0 -> 1030 bytes .../Icon/actions/document-properties.png | Bin 0 -> 767 bytes .../Umami/Icon/actions/document-save-as.png | Bin 0 -> 1048 bytes .../Umami/Icon/actions/document-save.png | Bin 0 -> 1109 bytes Media/Themes/Umami/Icon/actions/down.png | Bin 0 -> 1072 bytes .../Themes/Umami/Icon/actions/edit-clear.png | Bin 0 -> 1381 bytes Media/Themes/Umami/Icon/actions/edit-copy.png | Bin 0 -> 891 bytes Media/Themes/Umami/Icon/actions/edit-cut.png | Bin 0 -> 1272 bytes .../Themes/Umami/Icon/actions/edit-delete.png | Bin 0 -> 1389 bytes .../Umami/Icon/actions/edit-find-replace.png | Bin 0 -> 1244 bytes Media/Themes/Umami/Icon/actions/edit-find.png | Bin 0 -> 1254 bytes .../Themes/Umami/Icon/actions/edit-paste.png | Bin 0 -> 945 bytes Media/Themes/Umami/Icon/actions/edit-redo.png | Bin 0 -> 960 bytes .../Umami/Icon/actions/edit-select-all.png | Bin 0 -> 781 bytes Media/Themes/Umami/Icon/actions/edit-undo.png | Bin 0 -> 1161 bytes Media/Themes/Umami/Icon/actions/editclear.png | Bin 0 -> 1381 bytes Media/Themes/Umami/Icon/actions/editcopy.png | Bin 0 -> 891 bytes Media/Themes/Umami/Icon/actions/editcut.png | Bin 0 -> 1272 bytes .../Themes/Umami/Icon/actions/editdelete.png | Bin 0 -> 1389 bytes Media/Themes/Umami/Icon/actions/editpaste.png | Bin 0 -> 945 bytes Media/Themes/Umami/Icon/actions/exit.png | Bin 0 -> 1264 bytes Media/Themes/Umami/Icon/actions/filefind.png | Bin 0 -> 1254 bytes Media/Themes/Umami/Icon/actions/filenew.png | Bin 0 -> 851 bytes Media/Themes/Umami/Icon/actions/fileopen.png | Bin 0 -> 1093 bytes Media/Themes/Umami/Icon/actions/fileprint.png | Bin 0 -> 1030 bytes .../Umami/Icon/actions/filequickprint.png | Bin 0 -> 1291 bytes Media/Themes/Umami/Icon/actions/filesave.png | Bin 0 -> 1109 bytes .../Themes/Umami/Icon/actions/filesaveas.png | Bin 0 -> 1048 bytes Media/Themes/Umami/Icon/actions/find.png | Bin 0 -> 1254 bytes Media/Themes/Umami/Icon/actions/finish.png | Bin 0 -> 1106 bytes .../Themes/Umami/Icon/actions/folder-new.png | Bin 0 -> 1014 bytes .../Themes/Umami/Icon/actions/folder_new.png | Bin 0 -> 1014 bytes .../Umami/Icon/actions/format-indent-less.png | Bin 0 -> 840 bytes .../Umami/Icon/actions/format-indent-more.png | Bin 0 -> 854 bytes .../Icon/actions/format-justify-center.png | Bin 0 -> 694 bytes .../Icon/actions/format-justify-fill.png | Bin 0 -> 683 bytes .../Icon/actions/format-justify-left.png | Bin 0 -> 684 bytes .../Icon/actions/format-justify-right.png | Bin 0 -> 696 bytes .../Umami/Icon/actions/format-text-bold.png | Bin 0 -> 1109 bytes .../Umami/Icon/actions/format-text-italic.png | Bin 0 -> 952 bytes .../actions/format-text-strikethrough.png | Bin 0 -> 948 bytes .../Icon/actions/format-text-underline.png | Bin 0 -> 1038 bytes Media/Themes/Umami/Icon/actions/forward.png | Bin 0 -> 1100 bytes .../Umami/Icon/actions/gnome-lockscreen.png | Bin 0 -> 1169 bytes .../Umami/Icon/actions/gnome-logout.png | Bin 0 -> 1264 bytes .../Umami/Icon/actions/gnome-searchtool.png | Bin 0 -> 1414 bytes .../Icon/actions/gnome-session-logout.png | Bin 0 -> 1264 bytes .../Umami/Icon/actions/gnome-shutdown.png | Bin 0 -> 883 bytes .../Icon/actions/gnome-stock-mail-fwd.png | Bin 0 -> 985 bytes .../Icon/actions/gnome-stock-mail-new.png | Bin 0 -> 1102 bytes .../Icon/actions/gnome-stock-mail-rpl.png | Bin 0 -> 1181 bytes .../Icon/actions/gnome-stock-text-indent.png | Bin 0 -> 854 bytes .../actions/gnome-stock-text-unindent.png | Bin 0 -> 840 bytes Media/Themes/Umami/Icon/actions/go-bottom.png | Bin 0 -> 1111 bytes Media/Themes/Umami/Icon/actions/go-down.png | Bin 0 -> 1072 bytes Media/Themes/Umami/Icon/actions/go-first.png | Bin 0 -> 1140 bytes Media/Themes/Umami/Icon/actions/go-home.png | Bin 0 -> 1072 bytes Media/Themes/Umami/Icon/actions/go-jump.png | Bin 0 -> 1258 bytes Media/Themes/Umami/Icon/actions/go-last.png | Bin 0 -> 1106 bytes Media/Themes/Umami/Icon/actions/go-next.png | Bin 0 -> 1100 bytes .../Themes/Umami/Icon/actions/go-previous.png | Bin 0 -> 1135 bytes Media/Themes/Umami/Icon/actions/go-top.png | Bin 0 -> 1125 bytes Media/Themes/Umami/Icon/actions/go-up.png | Bin 0 -> 1063 bytes Media/Themes/Umami/Icon/actions/gohome.png | Bin 0 -> 1072 bytes Media/Themes/Umami/Icon/actions/gtk-add.png | Bin 0 -> 506 bytes Media/Themes/Umami/Icon/actions/gtk-bold.png | Bin 0 -> 1109 bytes .../Themes/Umami/Icon/actions/gtk-cancel.png | Bin 0 -> 1375 bytes Media/Themes/Umami/Icon/actions/gtk-clear.png | Bin 0 -> 1381 bytes Media/Themes/Umami/Icon/actions/gtk-copy.png | Bin 0 -> 891 bytes Media/Themes/Umami/Icon/actions/gtk-cut.png | Bin 0 -> 1272 bytes .../Themes/Umami/Icon/actions/gtk-delete.png | Bin 0 -> 1389 bytes .../Icon/actions/gtk-find-and-replace.png | Bin 0 -> 1244 bytes Media/Themes/Umami/Icon/actions/gtk-find.png | Bin 0 -> 1254 bytes .../Umami/Icon/actions/gtk-fullscreen.png | Bin 0 -> 988 bytes .../Umami/Icon/actions/gtk-go-back-ltr.png | Bin 0 -> 1135 bytes .../Umami/Icon/actions/gtk-go-back-rtl.png | Bin 0 -> 1100 bytes .../Themes/Umami/Icon/actions/gtk-go-down.png | Bin 0 -> 1072 bytes .../Umami/Icon/actions/gtk-go-forward-ltr.png | Bin 0 -> 1100 bytes .../Umami/Icon/actions/gtk-go-forward-rtl.png | Bin 0 -> 1135 bytes Media/Themes/Umami/Icon/actions/gtk-go-up.png | Bin 0 -> 1063 bytes .../Umami/Icon/actions/gtk-goto-bottom.png | Bin 0 -> 1111 bytes .../Umami/Icon/actions/gtk-goto-first-ltr.png | Bin 0 -> 1140 bytes .../Umami/Icon/actions/gtk-goto-first-rtl.png | Bin 0 -> 1106 bytes .../Umami/Icon/actions/gtk-goto-last-ltr.png | Bin 0 -> 1106 bytes .../Umami/Icon/actions/gtk-goto-last-rtl.png | Bin 0 -> 1140 bytes .../Umami/Icon/actions/gtk-goto-top.png | Bin 0 -> 1125 bytes Media/Themes/Umami/Icon/actions/gtk-home.png | Bin 0 -> 1072 bytes .../Umami/Icon/actions/gtk-indent-ltr.png | Bin 0 -> 854 bytes .../Umami/Icon/actions/gtk-indent-rtl.png | Bin 0 -> 840 bytes .../Themes/Umami/Icon/actions/gtk-italic.png | Bin 0 -> 952 bytes .../Umami/Icon/actions/gtk-jump-to-ltr.png | Bin 0 -> 1258 bytes .../Umami/Icon/actions/gtk-jump-to-rtl.png | Bin 0 -> 1258 bytes .../Umami/Icon/actions/gtk-justify-center.png | Bin 0 -> 694 bytes .../Umami/Icon/actions/gtk-justify-fill.png | Bin 0 -> 683 bytes .../Umami/Icon/actions/gtk-justify-left.png | Bin 0 -> 684 bytes .../Umami/Icon/actions/gtk-justify-right.png | Bin 0 -> 696 bytes .../Icon/actions/gtk-media-forward-ltr.png | Bin 0 -> 1229 bytes .../Icon/actions/gtk-media-forward-rtl.png | Bin 0 -> 1184 bytes .../Umami/Icon/actions/gtk-media-next-ltr.png | Bin 0 -> 1318 bytes .../Umami/Icon/actions/gtk-media-next-rtl.png | Bin 0 -> 1246 bytes .../Umami/Icon/actions/gtk-media-pause.png | Bin 0 -> 798 bytes .../Umami/Icon/actions/gtk-media-play-ltr.png | Bin 0 -> 1157 bytes .../Icon/actions/gtk-media-previous-ltr.png | Bin 0 -> 1246 bytes .../Icon/actions/gtk-media-previous-rtl.png | Bin 0 -> 1318 bytes .../Umami/Icon/actions/gtk-media-record.png | Bin 0 -> 1258 bytes .../Icon/actions/gtk-media-rewind-ltr.png | Bin 0 -> 1184 bytes .../Icon/actions/gtk-media-rewind-rtl.png | Bin 0 -> 1229 bytes .../Umami/Icon/actions/gtk-media-stop.png | Bin 0 -> 681 bytes Media/Themes/Umami/Icon/actions/gtk-new.png | Bin 0 -> 851 bytes Media/Themes/Umami/Icon/actions/gtk-open.png | Bin 0 -> 1093 bytes Media/Themes/Umami/Icon/actions/gtk-paste.png | Bin 0 -> 945 bytes .../Umami/Icon/actions/gtk-print-preview.png | Bin 0 -> 1291 bytes Media/Themes/Umami/Icon/actions/gtk-print.png | Bin 0 -> 1030 bytes .../Umami/Icon/actions/gtk-properties.png | Bin 0 -> 767 bytes .../Umami/Icon/actions/gtk-redo-ltr.png | Bin 0 -> 960 bytes .../Themes/Umami/Icon/actions/gtk-refresh.png | Bin 0 -> 1495 bytes .../Themes/Umami/Icon/actions/gtk-remove.png | Bin 0 -> 394 bytes .../Themes/Umami/Icon/actions/gtk-save-as.png | Bin 0 -> 1048 bytes Media/Themes/Umami/Icon/actions/gtk-save.png | Bin 0 -> 1109 bytes .../Umami/Icon/actions/gtk-select-all.png | Bin 0 -> 781 bytes Media/Themes/Umami/Icon/actions/gtk-stop.png | Bin 0 -> 1375 bytes .../Umami/Icon/actions/gtk-strikethrough.png | Bin 0 -> 948 bytes .../Umami/Icon/actions/gtk-underline.png | Bin 0 -> 1038 bytes .../Umami/Icon/actions/gtk-undo-ltr.png | Bin 0 -> 1161 bytes .../Umami/Icon/actions/gtk-unindent-ltr.png | Bin 0 -> 840 bytes .../Umami/Icon/actions/gtk-unindent-rtl.png | Bin 0 -> 854 bytes Media/Themes/Umami/Icon/actions/kfind.png | Bin 0 -> 1414 bytes Media/Themes/Umami/Icon/actions/kfm_home.png | Bin 0 -> 1072 bytes Media/Themes/Umami/Icon/actions/leftjust.png | Bin 0 -> 684 bytes Media/Themes/Umami/Icon/actions/list-add.png | Bin 0 -> 506 bytes .../Themes/Umami/Icon/actions/list-remove.png | Bin 0 -> 394 bytes Media/Themes/Umami/Icon/actions/lock.png | Bin 0 -> 1169 bytes .../Umami/Icon/actions/mail-forward.png | Bin 0 -> 985 bytes .../Umami/Icon/actions/mail-mark-junk.png | Bin 0 -> 1488 bytes .../Umami/Icon/actions/mail-mark-not-junk.png | Bin 0 -> 1395 bytes .../Umami/Icon/actions/mail-message-new.png | Bin 0 -> 1102 bytes .../Umami/Icon/actions/mail-reply-all.png | Bin 0 -> 1409 bytes .../Umami/Icon/actions/mail-reply-sender.png | Bin 0 -> 1181 bytes .../Umami/Icon/actions/mail-send-receive.png | Bin 0 -> 1111 bytes .../Umami/Icon/actions/mail_forward.png | Bin 0 -> 985 bytes Media/Themes/Umami/Icon/actions/mail_new.png | Bin 0 -> 1102 bytes .../Themes/Umami/Icon/actions/mail_reply.png | Bin 0 -> 1181 bytes .../Umami/Icon/actions/mail_replyall.png | Bin 0 -> 1409 bytes Media/Themes/Umami/Icon/actions/mail_spam.png | Bin 0 -> 1488 bytes .../Themes/Umami/Icon/actions/media-eject.png | Bin 0 -> 901 bytes .../Icon/actions/media-playback-pause.png | Bin 0 -> 798 bytes .../Icon/actions/media-playback-start.png | Bin 0 -> 1157 bytes .../Icon/actions/media-playback-stop.png | Bin 0 -> 681 bytes .../Umami/Icon/actions/media-record.png | Bin 0 -> 1258 bytes .../Icon/actions/media-seek-backward.png | Bin 0 -> 1184 bytes .../Umami/Icon/actions/media-seek-forward.png | Bin 0 -> 1229 bytes .../Icon/actions/media-skip-backward.png | Bin 0 -> 1246 bytes .../Umami/Icon/actions/media-skip-forward.png | Bin 0 -> 1318 bytes Media/Themes/Umami/Icon/actions/next.png | Bin 0 -> 1100 bytes .../Umami/Icon/actions/player_eject.png | Bin 0 -> 901 bytes .../Themes/Umami/Icon/actions/player_end.png | Bin 0 -> 1318 bytes .../Themes/Umami/Icon/actions/player_fwd.png | Bin 0 -> 1229 bytes .../Umami/Icon/actions/player_pause.png | Bin 0 -> 798 bytes .../Themes/Umami/Icon/actions/player_play.png | Bin 0 -> 1157 bytes .../Umami/Icon/actions/player_record.png | Bin 0 -> 1258 bytes .../Themes/Umami/Icon/actions/player_rew.png | Bin 0 -> 1184 bytes .../Umami/Icon/actions/player_start.png | Bin 0 -> 1246 bytes .../Themes/Umami/Icon/actions/player_stop.png | Bin 0 -> 681 bytes Media/Themes/Umami/Icon/actions/previous.png | Bin 0 -> 1135 bytes .../Umami/Icon/actions/process-stop.png | Bin 0 -> 1375 bytes .../Themes/Umami/Icon/actions/redhat-home.png | Bin 0 -> 1072 bytes Media/Themes/Umami/Icon/actions/redo.png | Bin 0 -> 960 bytes Media/Themes/Umami/Icon/actions/reload.png | Bin 0 -> 1495 bytes Media/Themes/Umami/Icon/actions/reload3.png | Bin 0 -> 1495 bytes .../Umami/Icon/actions/reload_all_tabs.png | Bin 0 -> 1495 bytes .../Themes/Umami/Icon/actions/reload_page.png | Bin 0 -> 1495 bytes Media/Themes/Umami/Icon/actions/remove.png | Bin 0 -> 394 bytes Media/Themes/Umami/Icon/actions/rightjust.png | Bin 0 -> 696 bytes Media/Themes/Umami/Icon/actions/search.png | Bin 0 -> 1414 bytes Media/Themes/Umami/Icon/actions/start.png | Bin 0 -> 1140 bytes .../Umami/Icon/actions/stock_add-bookmark.png | Bin 0 -> 1101 bytes .../Umami/Icon/actions/stock_bottom.png | Bin 0 -> 1111 bytes .../Themes/Umami/Icon/actions/stock_copy.png | Bin 0 -> 891 bytes Media/Themes/Umami/Icon/actions/stock_cut.png | Bin 0 -> 1272 bytes .../Umami/Icon/actions/stock_delete.png | Bin 0 -> 1389 bytes .../Themes/Umami/Icon/actions/stock_down.png | Bin 0 -> 1072 bytes .../Icon/actions/stock_file-properites.png | Bin 0 -> 767 bytes .../Themes/Umami/Icon/actions/stock_first.png | Bin 0 -> 1140 bytes .../Umami/Icon/actions/stock_fullscreen.png | Bin 0 -> 988 bytes .../Icon/actions/stock_help-add-bookmark.png | Bin 0 -> 1101 bytes .../Themes/Umami/Icon/actions/stock_home.png | Bin 0 -> 1072 bytes .../Themes/Umami/Icon/actions/stock_last.png | Bin 0 -> 1106 bytes .../Themes/Umami/Icon/actions/stock_left.png | Bin 0 -> 1135 bytes .../Umami/Icon/actions/stock_mail-compose.png | Bin 0 -> 1102 bytes .../Umami/Icon/actions/stock_mail-forward.png | Bin 0 -> 985 bytes .../Icon/actions/stock_mail-reply-to-all.png | Bin 0 -> 1409 bytes .../Umami/Icon/actions/stock_mail-reply.png | Bin 0 -> 1181 bytes .../Icon/actions/stock_mail-send-receive.png | Bin 0 -> 1111 bytes .../Umami/Icon/actions/stock_media-fwd.png | Bin 0 -> 1229 bytes .../Umami/Icon/actions/stock_media-next.png | Bin 0 -> 1318 bytes .../Umami/Icon/actions/stock_media-pause.png | Bin 0 -> 798 bytes .../Umami/Icon/actions/stock_media-play.png | Bin 0 -> 1157 bytes .../Umami/Icon/actions/stock_media-prev.png | Bin 0 -> 1246 bytes .../Umami/Icon/actions/stock_media-rec.png | Bin 0 -> 1258 bytes .../Umami/Icon/actions/stock_media-rew.png | Bin 0 -> 1184 bytes .../Umami/Icon/actions/stock_media-stop.png | Bin 0 -> 681 bytes .../Icon/actions/stock_new-address-book.png | Bin 0 -> 1083 bytes .../Icon/actions/stock_new-appointment.png | Bin 0 -> 1477 bytes .../Umami/Icon/actions/stock_new-bcard.png | Bin 0 -> 1032 bytes .../Umami/Icon/actions/stock_new-dir.png | Bin 0 -> 1014 bytes .../Umami/Icon/actions/stock_new-tab.png | Bin 0 -> 791 bytes .../Umami/Icon/actions/stock_new-text.png | Bin 0 -> 851 bytes .../Umami/Icon/actions/stock_new-window.png | Bin 0 -> 875 bytes .../Umami/Icon/actions/stock_not-spam.png | Bin 0 -> 1395 bytes .../Themes/Umami/Icon/actions/stock_paste.png | Bin 0 -> 945 bytes .../Icon/actions/stock_print-preview.png | Bin 0 -> 1291 bytes .../Themes/Umami/Icon/actions/stock_print.png | Bin 0 -> 1030 bytes .../Umami/Icon/actions/stock_properties.png | Bin 0 -> 767 bytes .../Themes/Umami/Icon/actions/stock_redo.png | Bin 0 -> 960 bytes .../Umami/Icon/actions/stock_refresh.png | Bin 0 -> 1495 bytes .../Themes/Umami/Icon/actions/stock_right.png | Bin 0 -> 1100 bytes .../Umami/Icon/actions/stock_save-as.png | Bin 0 -> 1048 bytes .../Themes/Umami/Icon/actions/stock_save.png | Bin 0 -> 1109 bytes .../Icon/actions/stock_search-and-replace.png | Bin 0 -> 1244 bytes .../Umami/Icon/actions/stock_search.png | Bin 0 -> 1254 bytes .../Umami/Icon/actions/stock_select-all.png | Bin 0 -> 781 bytes .../Themes/Umami/Icon/actions/stock_spam.png | Bin 0 -> 1488 bytes .../Themes/Umami/Icon/actions/stock_stop.png | Bin 0 -> 1375 bytes .../Icon/actions/stock_text-strikethrough.png | Bin 0 -> 948 bytes .../Umami/Icon/actions/stock_text_bold.png | Bin 0 -> 1109 bytes .../Umami/Icon/actions/stock_text_center.png | Bin 0 -> 694 bytes .../Umami/Icon/actions/stock_text_indent.png | Bin 0 -> 854 bytes .../Umami/Icon/actions/stock_text_italic.png | Bin 0 -> 952 bytes .../Umami/Icon/actions/stock_text_justify.png | Bin 0 -> 683 bytes .../Umami/Icon/actions/stock_text_left.png | Bin 0 -> 684 bytes .../Umami/Icon/actions/stock_text_right.png | Bin 0 -> 696 bytes .../Icon/actions/stock_text_underlined.png | Bin 0 -> 1038 bytes .../Icon/actions/stock_text_unindent.png | Bin 0 -> 840 bytes Media/Themes/Umami/Icon/actions/stock_top.png | Bin 0 -> 1125 bytes .../Themes/Umami/Icon/actions/stock_undo.png | Bin 0 -> 1161 bytes Media/Themes/Umami/Icon/actions/stock_up.png | Bin 0 -> 1063 bytes Media/Themes/Umami/Icon/actions/stop.png | Bin 0 -> 1375 bytes .../Umami/Icon/actions/system-lock-screen.png | Bin 0 -> 1169 bytes .../Umami/Icon/actions/system-log-out.png | Bin 0 -> 1264 bytes .../Umami/Icon/actions/system-search.png | Bin 0 -> 1414 bytes .../Umami/Icon/actions/system-shutdown.png | Bin 0 -> 883 bytes Media/Themes/Umami/Icon/actions/tab-new.png | Bin 0 -> 791 bytes Media/Themes/Umami/Icon/actions/tab_new.png | Bin 0 -> 791 bytes Media/Themes/Umami/Icon/actions/text_bold.png | Bin 0 -> 1109 bytes .../Themes/Umami/Icon/actions/text_italic.png | Bin 0 -> 952 bytes .../Themes/Umami/Icon/actions/text_strike.png | Bin 0 -> 948 bytes .../Themes/Umami/Icon/actions/text_under.png | Bin 0 -> 1038 bytes Media/Themes/Umami/Icon/actions/top.png | Bin 0 -> 1125 bytes Media/Themes/Umami/Icon/actions/undo.png | Bin 0 -> 1161 bytes Media/Themes/Umami/Icon/actions/up.png | Bin 0 -> 1063 bytes .../Umami/Icon/actions/view-fullscreen.png | Bin 0 -> 988 bytes .../Umami/Icon/actions/view-refresh.png | Bin 0 -> 1495 bytes .../Themes/Umami/Icon/actions/window-new.png | Bin 0 -> 875 bytes .../Umami/Icon/actions/window_fullscreen.png | Bin 0 -> 988 bytes .../Themes/Umami/Icon/actions/window_new.png | Bin 0 -> 875 bytes .../Umami/Icon/actions/xfce-system-lock.png | Bin 0 -> 1169 bytes Media/Themes/Umami/Icon/apps/access.png | Bin 0 -> 1171 bytes .../Icon/apps/accessibility-directory.png | Bin 0 -> 1171 bytes .../Icon/apps/accessories-calculator.png | Bin 0 -> 977 bytes .../Icon/apps/accessories-character-map.png | Bin 0 -> 1001 bytes .../Icon/apps/accessories-text-editor.png | Bin 0 -> 1007 bytes Media/Themes/Umami/Icon/apps/background.png | Bin 0 -> 1170 bytes Media/Themes/Umami/Icon/apps/browser.png | Bin 0 -> 1393 bytes Media/Themes/Umami/Icon/apps/calc.png | Bin 0 -> 977 bytes .../Umami/Icon/apps/config-language.png | Bin 0 -> 1124 bytes Media/Themes/Umami/Icon/apps/config-users.png | Bin 0 -> 1402 bytes Media/Themes/Umami/Icon/apps/date.png | Bin 0 -> 915 bytes Media/Themes/Umami/Icon/apps/email.png | Bin 0 -> 930 bytes Media/Themes/Umami/Icon/apps/file-manager.png | Bin 0 -> 559 bytes Media/Themes/Umami/Icon/apps/fonts.png | Bin 0 -> 1030 bytes .../Umami/Icon/apps/gnome-calculator.png | Bin 0 -> 977 bytes .../Umami/Icon/apps/gnome-character-map.png | Bin 0 -> 1001 bytes Media/Themes/Umami/Icon/apps/gnome-help.png | Bin 0 -> 1484 bytes .../Themes/Umami/Icon/apps/gnome-monitor.png | Bin 0 -> 1160 bytes .../Umami/Icon/apps/gnome-remote-desktop.png | Bin 0 -> 1387 bytes .../Themes/Umami/Icon/apps/gnome-session.png | Bin 0 -> 1139 bytes ...me-settings-accessibility-technologies.png | Bin 0 -> 1171 bytes .../Icon/apps/gnome-settings-background.png | Bin 0 -> 1170 bytes .../Umami/Icon/apps/gnome-settings-font.png | Bin 0 -> 1030 bytes .../Icon/apps/gnome-settings-keybindings.png | Bin 0 -> 1163 bytes .../Umami/Icon/apps/gnome-settings-theme.png | Bin 0 -> 1132 bytes .../Themes/Umami/Icon/apps/gnome-terminal.png | Bin 0 -> 1203 bytes .../Umami/Icon/apps/gnome-window-manager.png | Bin 0 -> 714 bytes Media/Themes/Umami/Icon/apps/gucharmap.png | Bin 0 -> 1001 bytes Media/Themes/Umami/Icon/apps/help-browser.png | Bin 0 -> 1484 bytes .../Umami/Icon/apps/internet-group-chat.png | Bin 0 -> 642 bytes .../Themes/Umami/Icon/apps/internet-mail.png | Bin 0 -> 930 bytes .../Umami/Icon/apps/internet-news-reader.png | Bin 0 -> 706 bytes .../Umami/Icon/apps/internet-web-browser.png | Bin 0 -> 1393 bytes Media/Themes/Umami/Icon/apps/kcalc.png | Bin 0 -> 977 bytes Media/Themes/Umami/Icon/apps/kcharselect.png | Bin 0 -> 1001 bytes Media/Themes/Umami/Icon/apps/kcmkwm.png | Bin 0 -> 714 bytes Media/Themes/Umami/Icon/apps/kedit.png | Bin 0 -> 1007 bytes Media/Themes/Umami/Icon/apps/key_bindings.png | Bin 0 -> 1163 bytes Media/Themes/Umami/Icon/apps/kfm.png | Bin 0 -> 559 bytes Media/Themes/Umami/Icon/apps/khelpcenter.png | Bin 0 -> 1484 bytes Media/Themes/Umami/Icon/apps/konsole.png | Bin 0 -> 1203 bytes Media/Themes/Umami/Icon/apps/krfb.png | Bin 0 -> 1387 bytes Media/Themes/Umami/Icon/apps/kscreensaver.png | Bin 0 -> 1116 bytes Media/Themes/Umami/Icon/apps/ksysguard.png | Bin 0 -> 1160 bytes Media/Themes/Umami/Icon/apps/kuser.png | Bin 0 -> 1402 bytes Media/Themes/Umami/Icon/apps/kwin.png | Bin 0 -> 714 bytes Media/Themes/Umami/Icon/apps/locale.png | Bin 0 -> 1124 bytes Media/Themes/Umami/Icon/apps/mail_generic.png | Bin 0 -> 930 bytes .../Umami/Icon/apps/office-calendar.png | Bin 0 -> 915 bytes Media/Themes/Umami/Icon/apps/openterm.png | Bin 0 -> 1203 bytes .../preferences-desktop-accessibility.png | Bin 0 -> 1171 bytes ...eferences-desktop-assistive-technology.png | Bin 0 -> 1233 bytes .../Icon/apps/preferences-desktop-font.png | Bin 0 -> 1030 bytes ...preferences-desktop-keyboard-shortcuts.png | Bin 0 -> 1163 bytes .../Icon/apps/preferences-desktop-locale.png | Bin 0 -> 1124 bytes .../apps/preferences-desktop-multimedia.png | Bin 0 -> 1043 bytes .../preferences-desktop-remote-desktop.png | Bin 0 -> 1387 bytes .../apps/preferences-desktop-screensaver.png | Bin 0 -> 1116 bytes .../Icon/apps/preferences-desktop-theme.png | Bin 0 -> 1132 bytes .../apps/preferences-desktop-wallpaper.png | Bin 0 -> 1170 bytes .../apps/preferences-system-network-proxy.png | Bin 0 -> 1358 bytes .../Icon/apps/preferences-system-session.png | Bin 0 -> 1139 bytes .../Icon/apps/preferences-system-windows.png | Bin 0 -> 714 bytes Media/Themes/Umami/Icon/apps/proxy-config.png | Bin 0 -> 1358 bytes Media/Themes/Umami/Icon/apps/proxy.png | Bin 0 -> 1358 bytes Media/Themes/Umami/Icon/apps/redhat-email.png | Bin 0 -> 930 bytes .../Umami/Icon/apps/redhat-filemanager.png | Bin 0 -> 559 bytes .../Umami/Icon/apps/redhat-web-browser.png | Bin 0 -> 1393 bytes Media/Themes/Umami/Icon/apps/screensaver.png | Bin 0 -> 1116 bytes Media/Themes/Umami/Icon/apps/stock_proxy.png | Bin 0 -> 1358 bytes Media/Themes/Umami/Icon/apps/style.png | Bin 0 -> 1132 bytes .../Themes/Umami/Icon/apps/susehelpcenter.png | Bin 0 -> 1484 bytes .../Umami/Icon/apps/system-config-users.png | Bin 0 -> 1402 bytes .../Umami/Icon/apps/system-file-manager.png | Bin 0 -> 559 bytes .../Umami/Icon/apps/system-installer.png | Bin 0 -> 1368 bytes .../Icon/apps/system-software-update.png | Bin 0 -> 1518 bytes Media/Themes/Umami/Icon/apps/system-users.png | Bin 0 -> 1402 bytes Media/Themes/Umami/Icon/apps/terminal.png | Bin 0 -> 1203 bytes Media/Themes/Umami/Icon/apps/text-editor.png | Bin 0 -> 1007 bytes .../Themes/Umami/Icon/apps/update-manager.png | Bin 0 -> 1518 bytes .../Icon/apps/utilities-system-monitor.png | Bin 0 -> 1160 bytes .../Umami/Icon/apps/utilities-terminal.png | Bin 0 -> 1203 bytes Media/Themes/Umami/Icon/apps/wallpaper.png | Bin 0 -> 1170 bytes Media/Themes/Umami/Icon/apps/web-browser.png | Bin 0 -> 1393 bytes Media/Themes/Umami/Icon/apps/xfcalendar.png | Bin 0 -> 915 bytes Media/Themes/Umami/Icon/apps/xfce-edit.png | Bin 0 -> 1007 bytes .../Umami/Icon/apps/xfce-filemanager.png | Bin 0 -> 559 bytes Media/Themes/Umami/Icon/apps/xfce-mail.png | Bin 0 -> 930 bytes Media/Themes/Umami/Icon/apps/xfce-man.png | Bin 0 -> 1484 bytes .../Themes/Umami/Icon/apps/xfce-terminal.png | Bin 0 -> 1203 bytes .../Themes/Umami/Icon/apps/xfce4-backdrop.png | Bin 0 -> 1170 bytes .../Themes/Umami/Icon/apps/xfce4-session.png | Bin 0 -> 1139 bytes Media/Themes/Umami/Icon/apps/xfce4-ui.png | Bin 0 -> 1132 bytes Media/Themes/Umami/Icon/apps/xfwm4.png | Bin 0 -> 714 bytes .../Icon/apps/ximian-evolution-calendar.png | Bin 0 -> 915 bytes Media/Themes/Umami/Icon/apps/xscreensaver.png | Bin 0 -> 1116 bytes Media/Themes/Umami/Icon/apps/zen-icon.png | Bin 0 -> 1518 bytes .../categories/applications-accessories.png | Bin 0 -> 1471 bytes .../categories/applications-development.png | Bin 0 -> 1489 bytes .../Icon/categories/applications-games.png | Bin 0 -> 1261 bytes .../Icon/categories/applications-graphics.png | Bin 0 -> 1055 bytes .../Icon/categories/applications-internet.png | Bin 0 -> 1532 bytes .../categories/applications-multimedia.png | Bin 0 -> 1428 bytes .../Icon/categories/applications-office.png | Bin 0 -> 1338 bytes .../Icon/categories/applications-other.png | Bin 0 -> 1130 bytes .../Icon/categories/applications-system.png | Bin 0 -> 1532 bytes .../Icon/categories/gnome-applications.png | Bin 0 -> 1338 bytes .../Icon/categories/gnome-control-center.png | Bin 0 -> 708 bytes .../Umami/Icon/categories/gnome-devel.png | Bin 0 -> 1489 bytes .../Umami/Icon/categories/gnome-globe.png | Bin 0 -> 1532 bytes .../Umami/Icon/categories/gnome-graphics.png | Bin 0 -> 1055 bytes .../Umami/Icon/categories/gnome-joystick.png | Bin 0 -> 1261 bytes .../Icon/categories/gnome-multimedia.png | Bin 0 -> 1428 bytes .../Umami/Icon/categories/gnome-other.png | Bin 0 -> 1130 bytes .../Umami/Icon/categories/gnome-settings.png | Bin 0 -> 708 bytes .../Umami/Icon/categories/gnome-system.png | Bin 0 -> 1532 bytes .../Umami/Icon/categories/gnome-util.png | Bin 0 -> 1471 bytes .../Umami/Icon/categories/gtk-preferences.png | Bin 0 -> 708 bytes .../categories/input_devices_settings.png | Bin 0 -> 1397 bytes .../Themes/Umami/Icon/categories/kcontrol.png | Bin 0 -> 708 bytes .../Icon/categories/package_development.png | Bin 0 -> 1489 bytes .../Umami/Icon/categories/package_games.png | Bin 0 -> 1261 bytes .../Icon/categories/package_graphics.png | Bin 0 -> 1055 bytes .../Icon/categories/package_multimedia.png | Bin 0 -> 1428 bytes .../Umami/Icon/categories/package_network.png | Bin 0 -> 1532 bytes .../Umami/Icon/categories/package_office.png | Bin 0 -> 1338 bytes .../Icon/categories/package_settings.png | Bin 0 -> 1143 bytes .../Umami/Icon/categories/package_system.png | Bin 0 -> 1532 bytes .../Icon/categories/package_utilities.png | Bin 0 -> 1471 bytes .../preferences-desktop-peripherals.png | Bin 0 -> 1397 bytes .../Icon/categories/preferences-desktop.png | Bin 0 -> 708 bytes .../Icon/categories/preferences-system.png | Bin 0 -> 1143 bytes .../Icon/categories/redhat-accessories.png | Bin 0 -> 1471 bytes .../Umami/Icon/categories/redhat-games.png | Bin 0 -> 1261 bytes .../Umami/Icon/categories/redhat-graphics.png | Bin 0 -> 1055 bytes .../Umami/Icon/categories/redhat-internet.png | Bin 0 -> 1532 bytes .../Umami/Icon/categories/redhat-office.png | Bin 0 -> 1338 bytes .../Icon/categories/redhat-preferences.png | Bin 0 -> 708 bytes .../Icon/categories/redhat-programming.png | Bin 0 -> 1489 bytes .../Icon/categories/redhat-sound_video.png | Bin 0 -> 1428 bytes .../categories/redhat-system_settings.png | Bin 0 -> 1143 bytes .../Icon/categories/redhat-system_tools.png | Bin 0 -> 1532 bytes .../Umami/Icon/categories/stock_internet.png | Bin 0 -> 1532 bytes .../Umami/Icon/categories/xfce-devel.png | Bin 0 -> 1489 bytes .../Umami/Icon/categories/xfce-games.png | Bin 0 -> 1261 bytes .../Umami/Icon/categories/xfce-graphics.png | Bin 0 -> 1055 bytes .../Umami/Icon/categories/xfce-internet.png | Bin 0 -> 1532 bytes .../Umami/Icon/categories/xfce-multimedia.png | Bin 0 -> 1428 bytes .../Umami/Icon/categories/xfce-office.png | Bin 0 -> 1338 bytes .../Icon/categories/xfce-system-settings.png | Bin 0 -> 1143 bytes .../Umami/Icon/categories/xfce-system.png | Bin 0 -> 1532 bytes .../Umami/Icon/categories/xfce-utils.png | Bin 0 -> 1471 bytes .../Umami/Icon/categories/xfce4-settings.png | Bin 0 -> 708 bytes Media/Themes/Umami/Icon/clipboard.png | Bin 0 -> 1196 bytes Media/Themes/Umami/Icon/default.png | Bin 0 -> 535 bytes .../Umami/Icon/devices/3floppy_unmount.png | Bin 0 -> 980 bytes .../Themes/Umami/Icon/devices/audio-card.png | Bin 0 -> 1319 bytes .../Icon/devices/audio-input-microphone.png | Bin 0 -> 1088 bytes Media/Themes/Umami/Icon/devices/battery.png | Bin 0 -> 1214 bytes .../Umami/Icon/devices/camera-photo.png | Bin 0 -> 1410 bytes .../Umami/Icon/devices/camera-video.png | Bin 0 -> 1169 bytes Media/Themes/Umami/Icon/devices/camera.png | Bin 0 -> 1410 bytes .../Umami/Icon/devices/camera_unmount.png | Bin 0 -> 1410 bytes .../Umami/Icon/devices/cdrom_unmount.png | Bin 0 -> 1507 bytes .../Umami/Icon/devices/cdwriter_unmount.png | Bin 0 -> 1507 bytes .../Themes/Umami/Icon/devices/chardevice.png | Bin 0 -> 1119 bytes Media/Themes/Umami/Icon/devices/computer.png | Bin 0 -> 1215 bytes Media/Themes/Umami/Icon/devices/display.png | Bin 0 -> 1119 bytes .../Themes/Umami/Icon/devices/drive-cdrom.png | Bin 0 -> 1176 bytes .../Umami/Icon/devices/drive-harddisk.png | Bin 0 -> 696 bytes .../Umami/Icon/devices/drive-optical.png | Bin 0 -> 1176 bytes .../Icon/devices/drive-removable-media.png | Bin 0 -> 775 bytes .../Themes/Umami/Icon/devices/dvd_unmount.png | Bin 0 -> 1507 bytes .../Umami/Icon/devices/gnome-dev-battery.png | Bin 0 -> 1214 bytes .../Icon/devices/gnome-dev-cdrom-audio.png | Bin 0 -> 1507 bytes .../Umami/Icon/devices/gnome-dev-cdrom.png | Bin 0 -> 1176 bytes .../Umami/Icon/devices/gnome-dev-computer.png | Bin 0 -> 1215 bytes .../Umami/Icon/devices/gnome-dev-disc-cdr.png | Bin 0 -> 1507 bytes .../Icon/devices/gnome-dev-disc-cdrw.png | Bin 0 -> 1507 bytes .../Icon/devices/gnome-dev-disc-dvdr-plus.png | Bin 0 -> 1507 bytes .../Icon/devices/gnome-dev-disc-dvdr.png | Bin 0 -> 1507 bytes .../Icon/devices/gnome-dev-disc-dvdram.png | Bin 0 -> 1507 bytes .../Icon/devices/gnome-dev-disc-dvdrom.png | Bin 0 -> 1507 bytes .../Icon/devices/gnome-dev-disc-dvdrw.png | Bin 0 -> 1507 bytes .../Umami/Icon/devices/gnome-dev-dvd.png | Bin 0 -> 1176 bytes .../Umami/Icon/devices/gnome-dev-ethernet.png | Bin 0 -> 1324 bytes .../Umami/Icon/devices/gnome-dev-floppy.png | Bin 0 -> 980 bytes .../Icon/devices/gnome-dev-harddisk-1394.png | Bin 0 -> 696 bytes .../Icon/devices/gnome-dev-harddisk-usb.png | Bin 0 -> 696 bytes .../Umami/Icon/devices/gnome-dev-harddisk.png | Bin 0 -> 696 bytes .../Umami/Icon/devices/gnome-dev-ipod.png | Bin 0 -> 1241 bytes .../Umami/Icon/devices/gnome-dev-keyboard.png | Bin 0 -> 890 bytes .../Umami/Icon/devices/gnome-dev-media-cf.png | Bin 0 -> 987 bytes .../Umami/Icon/devices/gnome-dev-media-ms.png | Bin 0 -> 987 bytes .../Icon/devices/gnome-dev-media-sdmmc.png | Bin 0 -> 987 bytes .../Umami/Icon/devices/gnome-dev-media-sm.png | Bin 0 -> 987 bytes .../Icon/devices/gnome-dev-mouse-ball.png | Bin 0 -> 1288 bytes .../Icon/devices/gnome-dev-mouse-optical.png | Bin 0 -> 1288 bytes .../Umami/Icon/devices/gnome-dev-printer.png | Bin 0 -> 979 bytes .../Icon/devices/gnome-dev-removable-1394.png | Bin 0 -> 775 bytes .../Icon/devices/gnome-dev-removable-usb.png | Bin 0 -> 775 bytes .../Icon/devices/gnome-dev-removable.png | Bin 0 -> 775 bytes .../Umami/Icon/devices/gnome-dev-wavelan.png | Bin 0 -> 1119 bytes .../Umami/Icon/devices/gnome-fs-client.png | Bin 0 -> 1215 bytes .../Umami/Icon/devices/gnome-stock-mic.png | Bin 0 -> 1088 bytes Media/Themes/Umami/Icon/devices/gtk-cdrom.png | Bin 0 -> 1507 bytes .../Themes/Umami/Icon/devices/gtk-floppy.png | Bin 0 -> 980 bytes .../Umami/Icon/devices/gtk-harddisk.png | Bin 0 -> 696 bytes Media/Themes/Umami/Icon/devices/harddrive.png | Bin 0 -> 696 bytes .../Themes/Umami/Icon/devices/hdd_unmount.png | Bin 0 -> 696 bytes .../Umami/Icon/devices/input-gaming.png | Bin 0 -> 1065 bytes .../Umami/Icon/devices/input-keyboard.png | Bin 0 -> 890 bytes .../Themes/Umami/Icon/devices/input-mouse.png | Bin 0 -> 1288 bytes .../Themes/Umami/Icon/devices/ipod_mount.png | Bin 0 -> 1241 bytes Media/Themes/Umami/Icon/devices/joystick.png | Bin 0 -> 1065 bytes Media/Themes/Umami/Icon/devices/keyboard.png | Bin 0 -> 890 bytes .../Themes/Umami/Icon/devices/kjobviewer.png | Bin 0 -> 979 bytes Media/Themes/Umami/Icon/devices/kxkb.png | Bin 0 -> 890 bytes .../Themes/Umami/Icon/devices/media-cdrom.png | Bin 0 -> 1507 bytes .../Themes/Umami/Icon/devices/media-flash.png | Bin 0 -> 987 bytes .../Umami/Icon/devices/media-floppy.png | Bin 0 -> 980 bytes .../Umami/Icon/devices/media-optical.png | Bin 0 -> 1507 bytes Media/Themes/Umami/Icon/devices/mouse.png | Bin 0 -> 1288 bytes .../Umami/Icon/devices/multimedia-player.png | Bin 0 -> 1241 bytes .../Umami/Icon/devices/network-wired.png | Bin 0 -> 1324 bytes .../Umami/Icon/devices/network-wireless.png | Bin 0 -> 1119 bytes Media/Themes/Umami/Icon/devices/nm-adhoc.png | Bin 0 -> 1119 bytes .../Umami/Icon/devices/nm-device-wired.png | Bin 0 -> 1324 bytes .../Umami/Icon/devices/nm-device-wireless.png | Bin 0 -> 1119 bytes .../Umami/Icon/devices/printer-remote.png | Bin 0 -> 979 bytes Media/Themes/Umami/Icon/devices/printer.png | Bin 0 -> 979 bytes Media/Themes/Umami/Icon/devices/printer1.png | Bin 0 -> 979 bytes Media/Themes/Umami/Icon/devices/printmgr.png | Bin 0 -> 979 bytes Media/Themes/Umami/Icon/devices/stock_mic.png | Bin 0 -> 1088 bytes .../Umami/Icon/devices/stock_printers.png | Bin 0 -> 979 bytes .../Umami/Icon/devices/system-floppy.png | Bin 0 -> 980 bytes Media/Themes/Umami/Icon/devices/system.png | Bin 0 -> 1215 bytes .../Icon/devices/usbpendrive_unmount.png | Bin 0 -> 775 bytes .../Umami/Icon/devices/video-display.png | Bin 0 -> 1119 bytes .../Umami/Icon/devices/xfce-printer.png | Bin 0 -> 979 bytes .../Umami/Icon/devices/xfce4-display.png | Bin 0 -> 1119 bytes .../Umami/Icon/devices/xfce4-keyboard.png | Bin 0 -> 890 bytes .../Themes/Umami/Icon/devices/xfce4-mouse.png | Bin 0 -> 1288 bytes Media/Themes/Umami/Icon/devices/yast_HD.png | Bin 0 -> 696 bytes .../Umami/Icon/devices/yast_idetude.png | Bin 0 -> 696 bytes .../Umami/Icon/devices/yast_joystick.png | Bin 0 -> 1065 bytes .../Themes/Umami/Icon/devices/yast_mouse.png | Bin 0 -> 1288 bytes .../Umami/Icon/devices/yast_printer.png | Bin 0 -> 979 bytes .../Umami/Icon/devices/yast_soundcard.png | Bin 0 -> 1319 bytes .../Umami/Icon/emblems/emblem-favorite.png | Bin 0 -> 1280 bytes .../Umami/Icon/emblems/emblem-important.png | Bin 0 -> 1254 bytes .../Umami/Icon/emblems/emblem-noread.png | Bin 0 -> 1060 bytes .../Umami/Icon/emblems/emblem-nowrite.png | Bin 0 -> 838 bytes .../Umami/Icon/emblems/emblem-photos.png | Bin 0 -> 1218 bytes .../Umami/Icon/emblems/emblem-readonly.png | Bin 0 -> 838 bytes .../Icon/emblems/emblem-symbolic-link.png | Bin 0 -> 904 bytes .../Umami/Icon/emblems/emblem-system.png | Bin 0 -> 1566 bytes .../Umami/Icon/emblems/emblem-unreadable.png | Bin 0 -> 1060 bytes Media/Themes/Umami/Icon/emotes/face-angel.png | Bin 0 -> 1496 bytes .../Themes/Umami/Icon/emotes/face-crying.png | Bin 0 -> 1490 bytes .../Umami/Icon/emotes/face-devilish.png | Bin 0 -> 1538 bytes .../Themes/Umami/Icon/emotes/face-glasses.png | Bin 0 -> 1567 bytes Media/Themes/Umami/Icon/emotes/face-grin.png | Bin 0 -> 1476 bytes Media/Themes/Umami/Icon/emotes/face-kiss.png | Bin 0 -> 1383 bytes .../Themes/Umami/Icon/emotes/face-monkey.png | Bin 0 -> 1411 bytes Media/Themes/Umami/Icon/emotes/face-plain.png | Bin 0 -> 1438 bytes Media/Themes/Umami/Icon/emotes/face-sad.png | Bin 0 -> 1468 bytes .../Umami/Icon/emotes/face-smile-big.png | Bin 0 -> 1455 bytes Media/Themes/Umami/Icon/emotes/face-smile.png | Bin 0 -> 1456 bytes .../Umami/Icon/emotes/face-surprise.png | Bin 0 -> 1482 bytes Media/Themes/Umami/Icon/emotes/face-wink.png | Bin 0 -> 1461 bytes .../Umami/Icon/emotes/stock_smiley-1.png | Bin 0 -> 1456 bytes .../Umami/Icon/emotes/stock_smiley-11.png | Bin 0 -> 1490 bytes .../Umami/Icon/emotes/stock_smiley-13.png | Bin 0 -> 1383 bytes .../Umami/Icon/emotes/stock_smiley-18.png | Bin 0 -> 1496 bytes .../Umami/Icon/emotes/stock_smiley-2.png | Bin 0 -> 1456 bytes .../Umami/Icon/emotes/stock_smiley-22.png | Bin 0 -> 1411 bytes .../Umami/Icon/emotes/stock_smiley-3.png | Bin 0 -> 1461 bytes .../Umami/Icon/emotes/stock_smiley-4.png | Bin 0 -> 1468 bytes .../Umami/Icon/emotes/stock_smiley-5.png | Bin 0 -> 1482 bytes .../Umami/Icon/emotes/stock_smiley-6.png | Bin 0 -> 1455 bytes .../Umami/Icon/emotes/stock_smiley-7.png | Bin 0 -> 1456 bytes .../Umami/Icon/emotes/stock_smiley-8.png | Bin 0 -> 1438 bytes .../mimetypes/application-certificate.png | Bin 0 -> 1440 bytes ...ion-vnd.ms-excel.sheet.macroEnabled.12.png | Bin 0 -> 1178 bytes ...owerpoint.presentation.macroEnabled.12.png | Bin 0 -> 1007 bytes ...n-vnd.ms-word.document.macroEnabled.12.png | Bin 0 -> 802 bytes ...cedocument.presentationml.presentation.png | Bin 0 -> 1007 bytes ...officedocument.presentationml.template.png | Bin 0 -> 1296 bytes ...ats-officedocument.spreadsheetml.sheet.png | Bin 0 -> 1178 bytes ...-officedocument.spreadsheetml.template.png | Bin 0 -> 1382 bytes ...ficedocument.wordprocessingml.document.png | Bin 0 -> 802 bytes ...ficedocument.wordprocessingml.template.png | Bin 0 -> 1190 bytes .../mimetypes/application-x-executable.png | Bin 0 -> 1198 bytes Media/Themes/Umami/Icon/mimetypes/ascii.png | Bin 0 -> 651 bytes .../Umami/Icon/mimetypes/audio-x-generic.png | Bin 0 -> 1124 bytes Media/Themes/Umami/Icon/mimetypes/binary.png | Bin 0 -> 1198 bytes .../Themes/Umami/Icon/mimetypes/contents2.png | Bin 0 -> 1019 bytes Media/Themes/Umami/Icon/mimetypes/deb.png | Bin 0 -> 847 bytes .../Themes/Umami/Icon/mimetypes/document.png | Bin 0 -> 802 bytes Media/Themes/Umami/Icon/mimetypes/empty.png | Bin 0 -> 651 bytes Media/Themes/Umami/Icon/mimetypes/exec.png | Bin 0 -> 1198 bytes .../Umami/Icon/mimetypes/folder_tar.png | Bin 0 -> 847 bytes .../Umami/Icon/mimetypes/font-x-generic.png | Bin 0 -> 1057 bytes Media/Themes/Umami/Icon/mimetypes/font.png | Bin 0 -> 1057 bytes .../Umami/Icon/mimetypes/font_bitmap.png | Bin 0 -> 1057 bytes .../Umami/Icon/mimetypes/font_truetype.png | Bin 0 -> 1057 bytes .../Umami/Icon/mimetypes/font_type1.png | Bin 0 -> 1057 bytes .../Icon/mimetypes/gnome-fs-executable.png | Bin 0 -> 1198 bytes .../gnome-mime-application-magicpoint.png | Bin 0 -> 1007 bytes .../gnome-mime-application-msword.png | Bin 0 -> 802 bytes .../mimetypes/gnome-mime-application-ogg.png | Bin 0 -> 1124 bytes .../mimetypes/gnome-mime-application-pdf.png | Bin 0 -> 802 bytes .../gnome-mime-application-postscript.png | Bin 0 -> 802 bytes .../mimetypes/gnome-mime-application-rtf.png | Bin 0 -> 802 bytes ...gnome-mime-application-vnd.lotus-1-2-3.png | Bin 0 -> 1178 bytes .../gnome-mime-application-vnd.ms-excel.png | Bin 0 -> 1178 bytes ...ome-mime-application-vnd.ms-powerpoint.png | Bin 0 -> 1007 bytes ...d.oasis.opendocument.graphics-template.png | Bin 0 -> 1247 bytes ...cation-vnd.oasis.opendocument.graphics.png | Bin 0 -> 883 bytes ...plication-vnd.oasis.opendocument.image.png | Bin 0 -> 883 bytes ...sis.opendocument.presentation-template.png | Bin 0 -> 1296 bytes ...on-vnd.oasis.opendocument.presentation.png | Bin 0 -> 1007 bytes ...asis.opendocument.spreadsheet-template.png | Bin 0 -> 1382 bytes ...ion-vnd.oasis.opendocument.spreadsheet.png | Bin 0 -> 1178 bytes ...n-vnd.oasis.opendocument.text-template.png | Bin 0 -> 1190 bytes ...cation-vnd.oasis.opendocument.text-web.png | Bin 0 -> 1256 bytes ...pplication-vnd.oasis.opendocument.text.png | Bin 0 -> 802 bytes ...me-application-vnd.rn-realmedia-secure.png | Bin 0 -> 1386 bytes ...-mime-application-vnd.rn-realmedia-vbr.png | Bin 0 -> 1386 bytes ...nome-mime-application-vnd.rn-realmedia.png | Bin 0 -> 1386 bytes ...mime-application-vnd.stardivision.calc.png | Bin 0 -> 1178 bytes ...e-application-vnd.stardivision.impress.png | Bin 0 -> 1007 bytes ...me-application-vnd.stardivision.writer.png | Bin 0 -> 802 bytes ...nome-mime-application-vnd.sun.xml.calc.png | Bin 0 -> 1178 bytes ...-application-vnd.sun.xml.calc.template.png | Bin 0 -> 1382 bytes ...nome-mime-application-vnd.sun.xml.draw.png | Bin 0 -> 883 bytes ...-application-vnd.sun.xml.draw.template.png | Bin 0 -> 1247 bytes ...e-mime-application-vnd.sun.xml.impress.png | Bin 0 -> 1007 bytes ...plication-vnd.sun.xml.impress.template.png | Bin 0 -> 1296 bytes ...me-mime-application-vnd.sun.xml.writer.png | Bin 0 -> 802 bytes ...pplication-vnd.sun.xml.writer.template.png | Bin 0 -> 1190 bytes .../gnome-mime-application-wordperfect.png | Bin 0 -> 802 bytes ...gnome-mime-application-x-7z-compressed.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-abiword.png | Bin 0 -> 802 bytes ...-mime-application-x-applix-spreadsheet.png | Bin 0 -> 1178 bytes .../gnome-mime-application-x-applix-word.png | Bin 0 -> 802 bytes .../gnome-mime-application-x-archive.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-arj.png | Bin 0 -> 847 bytes ...mime-application-x-bzip-compressed-tar.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-bzip.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-compress.png | Bin 0 -> 847 bytes ...nome-mime-application-x-compressed-tar.png | Bin 0 -> 847 bytes ...ome-mime-application-x-cpio-compressed.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-cpio.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-deb.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-dvi.png | Bin 0 -> 802 bytes .../gnome-mime-application-x-executable.png | Bin 0 -> 1198 bytes .../gnome-mime-application-x-font-afm.png | Bin 0 -> 1057 bytes .../gnome-mime-application-x-font-bdf.png | Bin 0 -> 1057 bytes ...nome-mime-application-x-font-linux-psf.png | Bin 0 -> 1057 bytes .../gnome-mime-application-x-font-pcf.png | Bin 0 -> 1057 bytes ...ome-mime-application-x-font-sunos-news.png | Bin 0 -> 1057 bytes .../gnome-mime-application-x-font-ttf.png | Bin 0 -> 1057 bytes .../gnome-mime-application-x-gnumeric.png | Bin 0 -> 1178 bytes .../gnome-mime-application-x-gzip.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-gzpostscript.png | Bin 0 -> 802 bytes .../gnome-mime-application-x-jar.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-killustrator.png | Bin 0 -> 1063 bytes .../gnome-mime-application-x-kpresenter.png | Bin 0 -> 1007 bytes .../gnome-mime-application-x-kspread.png | Bin 0 -> 1178 bytes .../gnome-mime-application-x-kword.png | Bin 0 -> 802 bytes .../gnome-mime-application-x-lha.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-lhz.png | Bin 0 -> 847 bytes ...mime-application-x-lzma-compressed-tar.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-lzma.png | Bin 0 -> 847 bytes ...e-mime-application-x-ms-dos-executable.png | Bin 0 -> 1198 bytes .../gnome-mime-application-x-perl.png | Bin 0 -> 1009 bytes .../gnome-mime-application-x-php.png | Bin 0 -> 1256 bytes ...ome-mime-application-x-python-bytecode.png | Bin 0 -> 1009 bytes .../gnome-mime-application-x-rar.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-rpm.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-scribus.png | Bin 0 -> 802 bytes .../gnome-mime-application-x-shellscript.png | Bin 0 -> 1009 bytes ...ome-mime-application-x-shockwave-flash.png | Bin 0 -> 1386 bytes .../gnome-mime-application-x-stuffit.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-tar.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-tarz.png | Bin 0 -> 847 bytes .../gnome-mime-application-x-tex.png | Bin 0 -> 802 bytes .../gnome-mime-application-xhtml+xml.png | Bin 0 -> 1256 bytes .../mimetypes/gnome-mime-application-zip.png | Bin 0 -> 847 bytes .../Umami/Icon/mimetypes/gnome-mime-audio.png | Bin 0 -> 1124 bytes .../Umami/Icon/mimetypes/gnome-mime-image.png | Bin 0 -> 1063 bytes .../Icon/mimetypes/gnome-mime-text-html.png | Bin 0 -> 1256 bytes .../mimetypes/gnome-mime-text-vnd.wap.wml.png | Bin 0 -> 1256 bytes .../Icon/mimetypes/gnome-mime-text-x-csh.png | Bin 0 -> 1009 bytes .../mimetypes/gnome-mime-text-x-python.png | Bin 0 -> 1009 bytes .../Icon/mimetypes/gnome-mime-text-x-sh.png | Bin 0 -> 1009 bytes .../mimetypes/gnome-mime-text-x-vcalendar.png | Bin 0 -> 1212 bytes .../mimetypes/gnome-mime-text-x-vcard.png | Bin 0 -> 1019 bytes .../Icon/mimetypes/gnome-mime-text-x-zsh.png | Bin 0 -> 1009 bytes .../Umami/Icon/mimetypes/gnome-mime-text.png | Bin 0 -> 651 bytes .../Umami/Icon/mimetypes/gnome-mime-video.png | Bin 0 -> 1386 bytes .../Icon/mimetypes/gnome-mime-x-font-afm.png | Bin 0 -> 1057 bytes .../Umami/Icon/mimetypes/gnome-package.png | Bin 0 -> 847 bytes Media/Themes/Umami/Icon/mimetypes/html.png | Bin 0 -> 1256 bytes .../Umami/Icon/mimetypes/image-x-generic.png | Bin 0 -> 1063 bytes Media/Themes/Umami/Icon/mimetypes/image.png | Bin 0 -> 1063 bytes .../Umami/Icon/mimetypes/kpresenter_kpr.png | Bin 0 -> 1007 bytes .../Umami/Icon/mimetypes/mime_ascii.png | Bin 0 -> 651 bytes Media/Themes/Umami/Icon/mimetypes/misc.png | Bin 0 -> 651 bytes .../Icon/mimetypes/package-x-generic.png | Bin 0 -> 847 bytes Media/Themes/Umami/Icon/mimetypes/package.png | Bin 0 -> 847 bytes .../Umami/Icon/mimetypes/package_editors.png | Bin 0 -> 651 bytes .../Icon/mimetypes/package_wordprocessing.png | Bin 0 -> 802 bytes Media/Themes/Umami/Icon/mimetypes/plan.png | Bin 0 -> 1212 bytes Media/Themes/Umami/Icon/mimetypes/rpm.png | Bin 0 -> 847 bytes .../Umami/Icon/mimetypes/shellscript.png | Bin 0 -> 1009 bytes Media/Themes/Umami/Icon/mimetypes/sound.png | Bin 0 -> 1124 bytes .../Umami/Icon/mimetypes/spreadsheet.png | Bin 0 -> 1178 bytes .../Icon/mimetypes/stock_addressbook.png | Bin 0 -> 1019 bytes .../Umami/Icon/mimetypes/stock_calendar.png | Bin 0 -> 1212 bytes .../Icon/mimetypes/stock_certificate.png | Bin 0 -> 1440 bytes .../Umami/Icon/mimetypes/stock_script.png | Bin 0 -> 1009 bytes Media/Themes/Umami/Icon/mimetypes/tar.png | Bin 0 -> 847 bytes .../Umami/Icon/mimetypes/template_source.png | Bin 0 -> 747 bytes .../Themes/Umami/Icon/mimetypes/text-html.png | Bin 0 -> 1256 bytes .../mimetypes/text-x-generic-template.png | Bin 0 -> 747 bytes .../Umami/Icon/mimetypes/text-x-generic.png | Bin 0 -> 651 bytes .../Umami/Icon/mimetypes/text-x-script.png | Bin 0 -> 1009 bytes Media/Themes/Umami/Icon/mimetypes/tgz.png | Bin 0 -> 847 bytes Media/Themes/Umami/Icon/mimetypes/txt.png | Bin 0 -> 651 bytes Media/Themes/Umami/Icon/mimetypes/txt2.png | Bin 0 -> 651 bytes Media/Themes/Umami/Icon/mimetypes/unknown.png | Bin 0 -> 651 bytes .../Themes/Umami/Icon/mimetypes/vcalendar.png | Bin 0 -> 1212 bytes Media/Themes/Umami/Icon/mimetypes/vcard.png | Bin 0 -> 1019 bytes .../Umami/Icon/mimetypes/video-x-generic.png | Bin 0 -> 1386 bytes Media/Themes/Umami/Icon/mimetypes/video.png | Bin 0 -> 1386 bytes .../Umami/Icon/mimetypes/wordprocessing.png | Bin 0 -> 802 bytes Media/Themes/Umami/Icon/mimetypes/www.png | Bin 0 -> 1256 bytes .../Icon/mimetypes/x-office-address-book.png | Bin 0 -> 1019 bytes .../Icon/mimetypes/x-office-calendar.png | Bin 0 -> 1212 bytes .../mimetypes/x-office-document-template.png | Bin 0 -> 1190 bytes .../Icon/mimetypes/x-office-document.png | Bin 0 -> 802 bytes .../mimetypes/x-office-drawing-template.png | Bin 0 -> 1247 bytes .../Umami/Icon/mimetypes/x-office-drawing.png | Bin 0 -> 883 bytes .../x-office-presentation-template.png | Bin 0 -> 1296 bytes .../Icon/mimetypes/x-office-presentation.png | Bin 0 -> 1007 bytes .../x-office-spreadsheet-template.png | Bin 0 -> 1382 bytes .../Icon/mimetypes/x-office-spreadsheet.png | Bin 0 -> 1178 bytes Media/Themes/Umami/Icon/mimetypes/zip.png | Bin 0 -> 847 bytes .../application-x-gnome-saved-search.png | Bin 0 -> 1207 bytes Media/Themes/Umami/Icon/places/desktop.png | Bin 0 -> 1109 bytes .../Umami/Icon/places/distributor-logo.png | Bin 0 -> 1119 bytes Media/Themes/Umami/Icon/places/emptytrash.png | Bin 0 -> 1323 bytes .../Umami/Icon/places/folder-remote.png | Bin 0 -> 1128 bytes .../Umami/Icon/places/folder-saved-search.png | Bin 0 -> 1207 bytes Media/Themes/Umami/Icon/places/folder.png | Bin 0 -> 1008 bytes .../Themes/Umami/Icon/places/folder_home.png | Bin 0 -> 1114 bytes .../Umami/Icon/places/gnome-fs-desktop.png | Bin 0 -> 1109 bytes .../Umami/Icon/places/gnome-fs-directory.png | Bin 0 -> 1008 bytes .../Themes/Umami/Icon/places/gnome-fs-ftp.png | Bin 0 -> 1128 bytes .../Umami/Icon/places/gnome-fs-home.png | Bin 0 -> 1114 bytes .../Umami/Icon/places/gnome-fs-network.png | Bin 0 -> 1071 bytes .../Themes/Umami/Icon/places/gnome-fs-nfs.png | Bin 0 -> 1128 bytes .../Umami/Icon/places/gnome-fs-server.png | Bin 0 -> 1021 bytes .../Umami/Icon/places/gnome-fs-share.png | Bin 0 -> 1128 bytes .../Themes/Umami/Icon/places/gnome-fs-smb.png | Bin 0 -> 1128 bytes .../Themes/Umami/Icon/places/gnome-fs-ssh.png | Bin 0 -> 1128 bytes .../Icon/places/gnome-fs-trash-empty.png | Bin 0 -> 1323 bytes .../Umami/Icon/places/gnome-main-menu.png | Bin 0 -> 1119 bytes .../gnome-mime-x-directory-nfs-server.png | Bin 0 -> 1021 bytes .../gnome-mime-x-directory-smb-server.png | Bin 0 -> 1021 bytes .../gnome-mime-x-directory-smb-share.png | Bin 0 -> 1128 bytes .../gnome-mime-x-directory-smb-workgroup.png | Bin 0 -> 1071 bytes .../Umami/Icon/places/gnome-stock-trash.png | Bin 0 -> 1323 bytes .../Umami/Icon/places/gtk-directory.png | Bin 0 -> 1008 bytes .../Themes/Umami/Icon/places/gtk-network.png | Bin 0 -> 1071 bytes .../Umami/Icon/places/inode-directory.png | Bin 0 -> 1008 bytes .../Umami/Icon/places/network-server.png | Bin 0 -> 1021 bytes .../Umami/Icon/places/network-workgroup.png | Bin 0 -> 1071 bytes Media/Themes/Umami/Icon/places/network.png | Bin 0 -> 1128 bytes .../Umami/Icon/places/network_local.png | Bin 0 -> 1071 bytes .../Umami/Icon/places/novell-button.png | Bin 0 -> 1119 bytes .../Icon/places/redhat-network-server.png | Bin 0 -> 1021 bytes Media/Themes/Umami/Icon/places/server.png | Bin 0 -> 1021 bytes Media/Themes/Umami/Icon/places/start-here.png | Bin 0 -> 1119 bytes .../Themes/Umami/Icon/places/stock_folder.png | Bin 0 -> 1008 bytes .../Umami/Icon/places/trashcan_empty.png | Bin 0 -> 1323 bytes .../Themes/Umami/Icon/places/user-desktop.png | Bin 0 -> 1109 bytes Media/Themes/Umami/Icon/places/user-home.png | Bin 0 -> 1114 bytes Media/Themes/Umami/Icon/places/user-trash.png | Bin 0 -> 1323 bytes .../Umami/Icon/places/xfce-trash_empty.png | Bin 0 -> 1323 bytes .../Umami/Icon/status/audio-volume-high.png | Bin 0 -> 1161 bytes .../Umami/Icon/status/audio-volume-low.png | Bin 0 -> 888 bytes .../Umami/Icon/status/audio-volume-medium.png | Bin 0 -> 1008 bytes .../Umami/Icon/status/audio-volume-muted.png | Bin 0 -> 1044 bytes .../Umami/Icon/status/battery-caution.png | Bin 0 -> 1244 bytes .../Umami/Icon/status/connect_creating.png | Bin 0 -> 906 bytes .../Umami/Icon/status/connect_established.png | Bin 0 -> 901 bytes Media/Themes/Umami/Icon/status/connect_no.png | Bin 0 -> 753 bytes .../Themes/Umami/Icon/status/dialog-error.png | Bin 0 -> 1075 bytes .../Umami/Icon/status/dialog-information.png | Bin 0 -> 1352 bytes .../Umami/Icon/status/dialog-warning.png | Bin 0 -> 1079 bytes Media/Themes/Umami/Icon/status/edittrash.png | Bin 0 -> 1589 bytes Media/Themes/Umami/Icon/status/error.png | Bin 0 -> 1075 bytes .../Umami/Icon/status/folder-drag-accept.png | Bin 0 -> 1041 bytes .../Themes/Umami/Icon/status/folder-open.png | Bin 0 -> 1073 bytes .../Umami/Icon/status/folder-visiting.png | Bin 0 -> 792 bytes .../Themes/Umami/Icon/status/folder_open.png | Bin 0 -> 1073 bytes .../status/gnome-dev-wavelan-encrypted.png | Bin 0 -> 1309 bytes .../Icon/status/gnome-fs-directory-accept.png | Bin 0 -> 1041 bytes .../status/gnome-fs-directory-visiting.png | Bin 0 -> 792 bytes .../Icon/status/gnome-fs-loading-icon.png | Bin 0 -> 955 bytes .../Umami/Icon/status/gnome-fs-trash-full.png | Bin 0 -> 1589 bytes .../Icon/status/gnome-netstatus-disconn.png | Bin 0 -> 753 bytes .../Icon/status/gnome-netstatus-error.png | Bin 0 -> 1138 bytes .../Icon/status/gnome-netstatus-idle.png | Bin 0 -> 901 bytes .../Umami/Icon/status/gnome-netstatus-rx.png | Bin 0 -> 1027 bytes .../Umami/Icon/status/gnome-netstatus-tx.png | Bin 0 -> 1030 bytes .../Icon/status/gnome-netstatus-txrx.png | Bin 0 -> 906 bytes .../Icon/status/gnome-stock-trash-full.png | Bin 0 -> 1589 bytes .../Umami/Icon/status/gtk-dialog-error.png | Bin 0 -> 1075 bytes .../Umami/Icon/status/gtk-dialog-info.png | Bin 0 -> 1352 bytes .../Umami/Icon/status/gtk-dialog-warning.png | Bin 0 -> 1079 bytes .../Umami/Icon/status/gtk-directory.png | Bin 0 -> 1073 bytes .../Umami/Icon/status/gtk-missing-image.png | Bin 0 -> 808 bytes .../Umami/Icon/status/image-loading.png | Bin 0 -> 955 bytes .../Umami/Icon/status/image-missing.png | Bin 0 -> 808 bytes Media/Themes/Umami/Icon/status/important.png | Bin 0 -> 1079 bytes Media/Themes/Umami/Icon/status/info.png | Bin 0 -> 1352 bytes .../Umami/Icon/status/mail-attachment.png | Bin 0 -> 1324 bytes .../Umami/Icon/status/messagebox_critical.png | Bin 0 -> 1075 bytes .../Icon/status/messagebox_critical_16x16.png | Bin 0 -> 1053 bytes .../Umami/Icon/status/messagebox_info.png | Bin 0 -> 1352 bytes .../Icon/status/messagebox_info_16x16.png | Bin 0 -> 1123 bytes .../Umami/Icon/status/messagebox_warning.png | Bin 0 -> 1079 bytes .../Icon/status/messagebox_warning_16x16.png | Bin 0 -> 1047 bytes .../Umami/Icon/status/network-error.png | Bin 0 -> 1138 bytes .../Themes/Umami/Icon/status/network-idle.png | Bin 0 -> 901 bytes .../Umami/Icon/status/network-offline.png | Bin 0 -> 753 bytes .../Umami/Icon/status/network-receive.png | Bin 0 -> 1027 bytes .../Icon/status/network-transmit-receive.png | Bin 0 -> 906 bytes .../Umami/Icon/status/network-transmit.png | Bin 0 -> 1030 bytes .../status/network-wireless-encrypted.png | Bin 0 -> 1309 bytes .../Umami/Icon/status/nm-no-connection.png | Bin 0 -> 753 bytes .../Umami/Icon/status/printer-error.png | Bin 0 -> 1169 bytes .../Icon/status/software-update-available.png | Bin 0 -> 1363 bytes .../Icon/status/software-update-urgent.png | Bin 0 -> 1036 bytes .../Themes/Umami/Icon/status/stock_attach.png | Bin 0 -> 1324 bytes .../Umami/Icon/status/stock_dialog-error.png | Bin 0 -> 1075 bytes .../Umami/Icon/status/stock_dialog-info.png | Bin 0 -> 1352 bytes .../Icon/status/stock_dialog-warning.png | Bin 0 -> 1079 bytes Media/Themes/Umami/Icon/status/stock_open.png | Bin 0 -> 1073 bytes .../Umami/Icon/status/stock_trash_full.png | Bin 0 -> 1589 bytes .../Umami/Icon/status/stock_volume-0.png | Bin 0 -> 888 bytes .../Umami/Icon/status/stock_volume-max.png | Bin 0 -> 1161 bytes .../Umami/Icon/status/stock_volume-med.png | Bin 0 -> 1008 bytes .../Umami/Icon/status/stock_volume-min.png | Bin 0 -> 888 bytes .../Umami/Icon/status/stock_volume-mute.png | Bin 0 -> 1044 bytes .../Themes/Umami/Icon/status/stock_volume.png | Bin 0 -> 1161 bytes .../Icon/status/stock_weather-cloudy.png | Bin 0 -> 1078 bytes .../Icon/status/stock_weather-few-clouds.png | Bin 0 -> 1277 bytes .../Icon/status/stock_weather-night-clear.png | Bin 0 -> 1079 bytes .../status/stock_weather-night-few-clouds.png | Bin 0 -> 1269 bytes .../Icon/status/stock_weather-showers.png | Bin 0 -> 1527 bytes .../Umami/Icon/status/stock_weather-snow.png | Bin 0 -> 1507 bytes .../Umami/Icon/status/stock_weather-storm.png | Bin 0 -> 1598 bytes .../Umami/Icon/status/stock_weather-sunny.png | Bin 0 -> 1079 bytes Media/Themes/Umami/Icon/status/sunny.png | Bin 0 -> 1079 bytes .../Umami/Icon/status/trashcan_full.png | Bin 0 -> 1589 bytes .../Umami/Icon/status/user-trash-full.png | Bin 0 -> 1589 bytes .../Umami/Icon/status/weather-clear-night.png | Bin 0 -> 1079 bytes .../Umami/Icon/status/weather-clear.png | Bin 0 -> 1079 bytes .../Icon/status/weather-few-clouds-night.png | Bin 0 -> 1269 bytes .../Umami/Icon/status/weather-few-clouds.png | Bin 0 -> 1277 bytes .../Umami/Icon/status/weather-overcast.png | Bin 0 -> 1078 bytes .../Icon/status/weather-severe-alert.png | Bin 0 -> 1440 bytes .../Icon/status/weather-showers-scattered.png | Bin 0 -> 1442 bytes .../Umami/Icon/status/weather-showers.png | Bin 0 -> 1527 bytes .../Themes/Umami/Icon/status/weather-snow.png | Bin 0 -> 1507 bytes .../Umami/Icon/status/weather-storm.png | Bin 0 -> 1598 bytes .../Umami/Icon/status/xfce-trash_full.png | Bin 0 -> 1589 bytes Media/Themes/Umami/Pointer/alternate.png | Bin 0 -> 922 bytes Media/Themes/Umami/Pointer/cross.png | Bin 0 -> 632 bytes Media/Themes/Umami/Pointer/dgn1.png | Bin 0 -> 951 bytes Media/Themes/Umami/Pointer/dgn2.png | Bin 0 -> 952 bytes Media/Themes/Umami/Pointer/help.png | Bin 0 -> 1346 bytes Media/Themes/Umami/Pointer/horz.png | Bin 0 -> 1064 bytes Media/Themes/Umami/Pointer/link.png | Bin 0 -> 999 bytes Media/Themes/Umami/Pointer/move.png | Bin 0 -> 1102 bytes Media/Themes/Umami/Pointer/pen.png | Bin 0 -> 1253 bytes Media/Themes/Umami/Pointer/pointer.png | Bin 0 -> 960 bytes Media/Themes/Umami/Pointer/text.png | Bin 0 -> 5514 bytes Media/Themes/Umami/Pointer/unavailable.png | Bin 0 -> 1314 bytes Media/Themes/Umami/Pointer/vert.png | Bin 0 -> 1084 bytes Media/Themes/Umami/Pointer/wait.png | Bin 0 -> 7107 bytes Media/Themes/Umami/Theme.HC | 1635 + Media/Themes/Umami/Widget/checkbox.png | Bin 0 -> 233 bytes Media/Themes/Umami/Widget/checkbox_active.png | Bin 0 -> 231 bytes .../Themes/Umami/Widget/checkbox_checked.png | Bin 0 -> 259 bytes .../Umami/Widget/checkbox_checked_active.png | Bin 0 -> 261 bytes Media/Themes/Umami/Widget/horz_slider.png | Bin 0 -> 224 bytes Media/Themes/Umami/Widget/radio.png | Bin 0 -> 255 bytes Media/Themes/Umami/Widget/radio_active.png | Bin 0 -> 258 bytes Media/Themes/Umami/Widget/radio_selected.png | Bin 0 -> 285 bytes Media/Themes/Umami/Widget/sb_down_button.png | Bin 0 -> 115 bytes Media/Themes/Umami/Widget/sb_left_button.png | Bin 0 -> 116 bytes Media/Themes/Umami/Widget/sb_right_button.png | Bin 0 -> 114 bytes Media/Themes/Umami/Widget/sb_up_button.png | Bin 0 -> 115 bytes Media/Themes/Umami/Widget/vert_slider.png | Bin 0 -> 162 bytes Media/Themes/Umami/Window/corner_resize.png | Bin 0 -> 1426 bytes Media/Themes/Umami/Window/min_max_close.png | Bin 0 -> 650 bytes .../Umami/Window/min_max_close_pressed.png | Bin 0 -> 359 bytes Media/Themes/Umami/wallpaper.jpg | Bin 0 -> 527141 bytes Media/Themes/Umami/wallpaper_old.png | Bin 0 -> 1839928 bytes Run.HC | 3 + Settings/SystemMenu.json | 40 + System/Api/Dns.HC | 28 + System/Api/Icmp.HC | 32 + System/Api/Ipv4.HC | 9 + System/Api/MD5.HC | 144 + System/Api/NetInfo.HC | 32 + System/Api/Tcp.HC | 210 + System/Api/Tls.HC | 99 + System/Config/Net.json | 10 + System/Core/Compositor.HC | 1112 + System/Core/FileSystem.HC | 3 + System/Core/Menu.HC | 67 + System/Core/MessageBox.HC | 108 + System/Core/Scheduler.HC | 79 + System/Core/Shell.HC | 282 + System/Core/ShellCommands.HC | 13 + System/Core/SystemStarter.HC | 214 + System/Core/SystemTray.HC | 139 + System/Drivers/AC97.HC | 153 + System/Drivers/Audio.HC | 117 + System/Drivers/Display.HC | 26 + System/Drivers/Mouse.HC | 111 + System/Drivers/Pci.HC | 98 + System/Drivers/VMSVGA.HC | 255 + System/Drivers/VMwareTools.HC | 148 + System/Drivers/Virtio-blk.HC | 475 + System/FFI/Base.HC | 51 + System/FFI/ELF64.HC | 301 + System/FFI/LibC.HC | 324 + System/FFI/New.HC | 35 + System/Jakt/DC.HC | 288 + System/Jakt/IOPort.HC | 53 + System/Jakt/Input.HC | 72 + System/Jakt/OS.HC | 289 + System/Jakt/PCI.HC | 62 + System/Jakt/Time.HC | 37 + System/Jakt/Window.HC | 94 + System/Libraries/Animation2D.HC | 65 + System/Libraries/Audio.HC | 143 + System/Libraries/Base64.HC | 90 + System/Libraries/BitmapFont.HC | 146 + System/Libraries/Clipboard.HC | 125 + System/Libraries/Display.HC | 49 + System/Libraries/FileSystem.HC | 234 + System/Libraries/Function.HC | 28 + System/Libraries/Graphics2D.HC | 1428 + System/Libraries/Gui.HC | 398 + System/Libraries/Http.HC | 628 + System/Libraries/Image.HC | 55 + System/Libraries/Ipc.HC | 54 + System/Libraries/Json.HC | 1550 + System/Libraries/RawText.HC | 104 + System/Libraries/Rsa.HC | 46 + System/Libraries/Session.HC | 8 + System/Libraries/Sha256.HC | 235 + System/Libraries/Shell.HC | 52 + System/Libraries/Stdio.HC | 155 + System/Libraries/String.HC | 171 + System/Libraries/System.HC | 104 + System/Libraries/Theme.HC | 49 + System/Libraries/Tlse.HC | 115 + System/Libraries/User.HC | 10 + System/Libraries/Widget.HC | 834 + System/MakeSystem.HC | 103 + System/Setup/Environment.HC | 229 + System/Setup/Init.HC | 65 + System/Shell/Commands/aplay.HC | 35 + System/Shell/Commands/cat.HC | 19 + System/Shell/Commands/clear.HC | 9 + System/Shell/Commands/echo.HC | 10 + System/Shell/Commands/esh.HC | 5 + System/Shell/Commands/history.HC | 14 + System/Shell/Commands/ifconfig.HC | 47 + System/Shell/Commands/ls.HC | 70 + System/Shell/Commands/nslookup.HC | 32 + System/Shell/Commands/open.HC | 8 + System/Shell/Commands/ping.HC | 58 + System/Shell/Commands/poweroff.HC | 5 + System/Shell/Commands/pwd.HC | 7 + System/Shell/Commands/reboot.HC | 5 + System/Shell/Commands/sh.HC | 4 + System/Shell/Commands/uname.HC | 81 + System/Shell/Commands/whoami.HC | 28 + System/Shell/Commands/wpset.HC | 24 + System/Utilities/Dns.HC | 10 + System/Utilities/Image.HC | 414 + System/Utilities/NetRep.HC | 22 + System/Utilities/Ping.HC | 72 + System/Utilities/Time.HC | 92 + scripts/build-all | 191 + src/image/image.c | 209 + src/image/stb_image.h | 8634 ++++ src/image/stb_image_write.h | 1807 + src/libtemple/ioport.h | 6 + src/libtemple/libtemple.cpp | 77 + src/libtemple/os.h | 15 + src/libtemple/pci.h | 10 + src/libtemple/time.h | 4 + src/net/devices/virtio.h | 28 + src/net/devices/virtio.jakt | 143 + src/net/lib/json.jakt | 350 + src/net/lib/util.jakt | 147 + src/net/net.jakt | 173 + src/net/os/ioport.jakt | 29 + src/net/os/os.jakt | 154 + src/net/os/pci.jakt | 103 + src/net/os/time.jakt | 94 + src/net/tcpip.jakt | 1675 + src/tlse/LICENSE | 60 + src/tlse/libtomcrypt.c | 34766 ++++++++++++++++ src/tlse/tlse.c | 12374 ++++++ src/tlse/tlse.h | 472 + 1029 files changed, 78311 insertions(+) create mode 100644 .clang-format create mode 100644 .gitattributes create mode 100644 .gitignore create mode 100644 .vscode/settings.json create mode 100644 .vscode/tasks.json create mode 100644 Applets/Calendar.applet/Run.HC create mode 100644 Applets/ClipManager.applet/Run.HC create mode 100644 Applets/NetworkStatus.applet/Run.HC create mode 100644 Applets/NetworkStatus.applet/window_icon_16x16.png create mode 100644 Applets/VolumeSlider.applet/Run.HC create mode 100644 Applications/Accessories/Calculator.app/Run.HC create mode 100644 Applications/Accessories/Calculator.app/window_icon_16x16.png create mode 100644 Applications/Accessories/Icon.png create mode 100644 Applications/OS/Icon.png create mode 100644 Applications/OS/MenuBar.app/Run.HC create mode 100644 Applications/OS/MenuBar.app/system_menu.png create mode 100644 Applications/OS/ShutDown.app/Icon.png create mode 100644 Applications/OS/ShutDown.app/Run.HC create mode 100644 Applications/OS/TaskSwitcher.app/Run.HC create mode 100644 Applications/OS/TempleOS.app/Run.HC create mode 100644 Applications/OS/TempleOS.app/window_icon_16x16.png create mode 100644 Applications/OS/Terminal.app/Icon.png create mode 100644 Applications/OS/Terminal.app/Run.HC create mode 100644 Applications/OS/Wallpaper.app/Run.HC create mode 100644 Applications/TestApplication.app/Run.HC create mode 100644 Include/Gui.HC create mode 100644 Media/Sounds/Beep.wav create mode 100644 Media/Themes/Umami/BitmapFont/menu.png create mode 100644 Media/Themes/Umami/BitmapFont/monospace.png create mode 100644 Media/Themes/Umami/BitmapFont/sans.png create mode 100644 Media/Themes/Umami/Icon/actions/add.png create mode 100644 Media/Themes/Umami/Icon/actions/address-book-new.png create mode 100644 Media/Themes/Umami/Icon/actions/appointment-new.png create mode 100644 Media/Themes/Umami/Icon/actions/appointment.png create mode 100644 Media/Themes/Umami/Icon/actions/back.png create mode 100644 Media/Themes/Umami/Icon/actions/bookmark-new.png create mode 100644 Media/Themes/Umami/Icon/actions/bookmark_add.png create mode 100644 Media/Themes/Umami/Icon/actions/bookmarks_list_add.png create mode 100644 Media/Themes/Umami/Icon/actions/bottom.png create mode 100644 Media/Themes/Umami/Icon/actions/centrejust.png create mode 100644 Media/Themes/Umami/Icon/actions/contact-new.png create mode 100644 Media/Themes/Umami/Icon/actions/document-new.png create mode 100644 Media/Themes/Umami/Icon/actions/document-open.png create mode 100644 Media/Themes/Umami/Icon/actions/document-print-preview.png create mode 100644 Media/Themes/Umami/Icon/actions/document-print.png create mode 100644 Media/Themes/Umami/Icon/actions/document-properties.png create mode 100644 Media/Themes/Umami/Icon/actions/document-save-as.png create mode 100644 Media/Themes/Umami/Icon/actions/document-save.png create mode 100644 Media/Themes/Umami/Icon/actions/down.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-clear.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-copy.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-cut.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-delete.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-find-replace.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-find.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-paste.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-redo.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-select-all.png create mode 100644 Media/Themes/Umami/Icon/actions/edit-undo.png create mode 100644 Media/Themes/Umami/Icon/actions/editclear.png create mode 100644 Media/Themes/Umami/Icon/actions/editcopy.png create mode 100644 Media/Themes/Umami/Icon/actions/editcut.png create mode 100644 Media/Themes/Umami/Icon/actions/editdelete.png create mode 100644 Media/Themes/Umami/Icon/actions/editpaste.png create mode 100644 Media/Themes/Umami/Icon/actions/exit.png create mode 100644 Media/Themes/Umami/Icon/actions/filefind.png create mode 100644 Media/Themes/Umami/Icon/actions/filenew.png create mode 100644 Media/Themes/Umami/Icon/actions/fileopen.png create mode 100644 Media/Themes/Umami/Icon/actions/fileprint.png create mode 100644 Media/Themes/Umami/Icon/actions/filequickprint.png create mode 100644 Media/Themes/Umami/Icon/actions/filesave.png create mode 100644 Media/Themes/Umami/Icon/actions/filesaveas.png create mode 100644 Media/Themes/Umami/Icon/actions/find.png create mode 100644 Media/Themes/Umami/Icon/actions/finish.png create mode 100644 Media/Themes/Umami/Icon/actions/folder-new.png create mode 100644 Media/Themes/Umami/Icon/actions/folder_new.png create mode 100644 Media/Themes/Umami/Icon/actions/format-indent-less.png create mode 100644 Media/Themes/Umami/Icon/actions/format-indent-more.png create mode 100644 Media/Themes/Umami/Icon/actions/format-justify-center.png create mode 100644 Media/Themes/Umami/Icon/actions/format-justify-fill.png create mode 100644 Media/Themes/Umami/Icon/actions/format-justify-left.png create mode 100644 Media/Themes/Umami/Icon/actions/format-justify-right.png create mode 100644 Media/Themes/Umami/Icon/actions/format-text-bold.png create mode 100644 Media/Themes/Umami/Icon/actions/format-text-italic.png create mode 100644 Media/Themes/Umami/Icon/actions/format-text-strikethrough.png create mode 100644 Media/Themes/Umami/Icon/actions/format-text-underline.png create mode 100644 Media/Themes/Umami/Icon/actions/forward.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-lockscreen.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-logout.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-searchtool.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-session-logout.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-shutdown.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-stock-mail-fwd.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-stock-mail-new.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-stock-mail-rpl.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-stock-text-indent.png create mode 100644 Media/Themes/Umami/Icon/actions/gnome-stock-text-unindent.png create mode 100644 Media/Themes/Umami/Icon/actions/go-bottom.png create mode 100644 Media/Themes/Umami/Icon/actions/go-down.png create mode 100644 Media/Themes/Umami/Icon/actions/go-first.png create mode 100644 Media/Themes/Umami/Icon/actions/go-home.png create mode 100644 Media/Themes/Umami/Icon/actions/go-jump.png create mode 100644 Media/Themes/Umami/Icon/actions/go-last.png create mode 100644 Media/Themes/Umami/Icon/actions/go-next.png create mode 100644 Media/Themes/Umami/Icon/actions/go-previous.png create mode 100644 Media/Themes/Umami/Icon/actions/go-top.png create mode 100644 Media/Themes/Umami/Icon/actions/go-up.png create mode 100644 Media/Themes/Umami/Icon/actions/gohome.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-add.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-bold.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-cancel.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-clear.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-copy.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-cut.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-delete.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-find-and-replace.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-find.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-fullscreen.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-go-back-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-go-back-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-go-down.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-go-forward-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-go-forward-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-go-up.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-goto-bottom.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-goto-first-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-goto-first-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-goto-last-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-goto-last-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-goto-top.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-home.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-indent-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-indent-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-italic.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-jump-to-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-jump-to-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-justify-center.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-justify-fill.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-justify-left.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-justify-right.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-forward-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-forward-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-next-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-next-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-pause.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-play-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-previous-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-previous-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-record.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-rewind-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-rewind-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-media-stop.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-new.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-open.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-paste.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-print-preview.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-print.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-properties.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-redo-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-refresh.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-remove.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-save-as.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-save.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-select-all.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-stop.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-strikethrough.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-underline.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-undo-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-unindent-ltr.png create mode 100644 Media/Themes/Umami/Icon/actions/gtk-unindent-rtl.png create mode 100644 Media/Themes/Umami/Icon/actions/kfind.png create mode 100644 Media/Themes/Umami/Icon/actions/kfm_home.png create mode 100644 Media/Themes/Umami/Icon/actions/leftjust.png create mode 100644 Media/Themes/Umami/Icon/actions/list-add.png create mode 100644 Media/Themes/Umami/Icon/actions/list-remove.png create mode 100644 Media/Themes/Umami/Icon/actions/lock.png create mode 100644 Media/Themes/Umami/Icon/actions/mail-forward.png create mode 100644 Media/Themes/Umami/Icon/actions/mail-mark-junk.png create mode 100644 Media/Themes/Umami/Icon/actions/mail-mark-not-junk.png create mode 100644 Media/Themes/Umami/Icon/actions/mail-message-new.png create mode 100644 Media/Themes/Umami/Icon/actions/mail-reply-all.png create mode 100644 Media/Themes/Umami/Icon/actions/mail-reply-sender.png create mode 100644 Media/Themes/Umami/Icon/actions/mail-send-receive.png create mode 100644 Media/Themes/Umami/Icon/actions/mail_forward.png create mode 100644 Media/Themes/Umami/Icon/actions/mail_new.png create mode 100644 Media/Themes/Umami/Icon/actions/mail_reply.png create mode 100644 Media/Themes/Umami/Icon/actions/mail_replyall.png create mode 100644 Media/Themes/Umami/Icon/actions/mail_spam.png create mode 100644 Media/Themes/Umami/Icon/actions/media-eject.png create mode 100644 Media/Themes/Umami/Icon/actions/media-playback-pause.png create mode 100644 Media/Themes/Umami/Icon/actions/media-playback-start.png create mode 100644 Media/Themes/Umami/Icon/actions/media-playback-stop.png create mode 100644 Media/Themes/Umami/Icon/actions/media-record.png create mode 100644 Media/Themes/Umami/Icon/actions/media-seek-backward.png create mode 100644 Media/Themes/Umami/Icon/actions/media-seek-forward.png create mode 100644 Media/Themes/Umami/Icon/actions/media-skip-backward.png create mode 100644 Media/Themes/Umami/Icon/actions/media-skip-forward.png create mode 100644 Media/Themes/Umami/Icon/actions/next.png create mode 100644 Media/Themes/Umami/Icon/actions/player_eject.png create mode 100644 Media/Themes/Umami/Icon/actions/player_end.png create mode 100644 Media/Themes/Umami/Icon/actions/player_fwd.png create mode 100644 Media/Themes/Umami/Icon/actions/player_pause.png create mode 100644 Media/Themes/Umami/Icon/actions/player_play.png create mode 100644 Media/Themes/Umami/Icon/actions/player_record.png create mode 100644 Media/Themes/Umami/Icon/actions/player_rew.png create mode 100644 Media/Themes/Umami/Icon/actions/player_start.png create mode 100644 Media/Themes/Umami/Icon/actions/player_stop.png create mode 100644 Media/Themes/Umami/Icon/actions/previous.png create mode 100644 Media/Themes/Umami/Icon/actions/process-stop.png create mode 100644 Media/Themes/Umami/Icon/actions/redhat-home.png create mode 100644 Media/Themes/Umami/Icon/actions/redo.png create mode 100644 Media/Themes/Umami/Icon/actions/reload.png create mode 100644 Media/Themes/Umami/Icon/actions/reload3.png create mode 100644 Media/Themes/Umami/Icon/actions/reload_all_tabs.png create mode 100644 Media/Themes/Umami/Icon/actions/reload_page.png create mode 100644 Media/Themes/Umami/Icon/actions/remove.png create mode 100644 Media/Themes/Umami/Icon/actions/rightjust.png create mode 100644 Media/Themes/Umami/Icon/actions/search.png create mode 100644 Media/Themes/Umami/Icon/actions/start.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_add-bookmark.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_bottom.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_copy.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_cut.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_delete.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_down.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_file-properites.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_first.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_fullscreen.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_help-add-bookmark.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_home.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_last.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_left.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_mail-compose.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_mail-forward.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_mail-reply-to-all.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_mail-reply.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_mail-send-receive.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_media-fwd.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_media-next.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_media-pause.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_media-play.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_media-prev.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_media-rec.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_media-rew.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_media-stop.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_new-address-book.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_new-appointment.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_new-bcard.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_new-dir.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_new-tab.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_new-text.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_new-window.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_not-spam.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_paste.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_print-preview.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_print.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_properties.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_redo.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_refresh.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_right.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_save-as.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_save.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_search-and-replace.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_search.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_select-all.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_spam.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_stop.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text-strikethrough.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_bold.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_center.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_indent.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_italic.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_justify.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_left.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_right.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_underlined.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_text_unindent.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_top.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_undo.png create mode 100644 Media/Themes/Umami/Icon/actions/stock_up.png create mode 100644 Media/Themes/Umami/Icon/actions/stop.png create mode 100644 Media/Themes/Umami/Icon/actions/system-lock-screen.png create mode 100644 Media/Themes/Umami/Icon/actions/system-log-out.png create mode 100644 Media/Themes/Umami/Icon/actions/system-search.png create mode 100644 Media/Themes/Umami/Icon/actions/system-shutdown.png create mode 100644 Media/Themes/Umami/Icon/actions/tab-new.png create mode 100644 Media/Themes/Umami/Icon/actions/tab_new.png create mode 100644 Media/Themes/Umami/Icon/actions/text_bold.png create mode 100644 Media/Themes/Umami/Icon/actions/text_italic.png create mode 100644 Media/Themes/Umami/Icon/actions/text_strike.png create mode 100644 Media/Themes/Umami/Icon/actions/text_under.png create mode 100644 Media/Themes/Umami/Icon/actions/top.png create mode 100644 Media/Themes/Umami/Icon/actions/undo.png create mode 100644 Media/Themes/Umami/Icon/actions/up.png create mode 100644 Media/Themes/Umami/Icon/actions/view-fullscreen.png create mode 100644 Media/Themes/Umami/Icon/actions/view-refresh.png create mode 100644 Media/Themes/Umami/Icon/actions/window-new.png create mode 100644 Media/Themes/Umami/Icon/actions/window_fullscreen.png create mode 100644 Media/Themes/Umami/Icon/actions/window_new.png create mode 100644 Media/Themes/Umami/Icon/actions/xfce-system-lock.png create mode 100644 Media/Themes/Umami/Icon/apps/access.png create mode 100644 Media/Themes/Umami/Icon/apps/accessibility-directory.png create mode 100644 Media/Themes/Umami/Icon/apps/accessories-calculator.png create mode 100644 Media/Themes/Umami/Icon/apps/accessories-character-map.png create mode 100644 Media/Themes/Umami/Icon/apps/accessories-text-editor.png create mode 100644 Media/Themes/Umami/Icon/apps/background.png create mode 100644 Media/Themes/Umami/Icon/apps/browser.png create mode 100644 Media/Themes/Umami/Icon/apps/calc.png create mode 100644 Media/Themes/Umami/Icon/apps/config-language.png create mode 100644 Media/Themes/Umami/Icon/apps/config-users.png create mode 100644 Media/Themes/Umami/Icon/apps/date.png create mode 100644 Media/Themes/Umami/Icon/apps/email.png create mode 100644 Media/Themes/Umami/Icon/apps/file-manager.png create mode 100644 Media/Themes/Umami/Icon/apps/fonts.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-calculator.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-character-map.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-help.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-monitor.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-remote-desktop.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-session.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-settings-accessibility-technologies.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-settings-background.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-settings-font.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-settings-keybindings.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-settings-theme.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-terminal.png create mode 100644 Media/Themes/Umami/Icon/apps/gnome-window-manager.png create mode 100644 Media/Themes/Umami/Icon/apps/gucharmap.png create mode 100644 Media/Themes/Umami/Icon/apps/help-browser.png create mode 100644 Media/Themes/Umami/Icon/apps/internet-group-chat.png create mode 100644 Media/Themes/Umami/Icon/apps/internet-mail.png create mode 100644 Media/Themes/Umami/Icon/apps/internet-news-reader.png create mode 100644 Media/Themes/Umami/Icon/apps/internet-web-browser.png create mode 100644 Media/Themes/Umami/Icon/apps/kcalc.png create mode 100644 Media/Themes/Umami/Icon/apps/kcharselect.png create mode 100644 Media/Themes/Umami/Icon/apps/kcmkwm.png create mode 100644 Media/Themes/Umami/Icon/apps/kedit.png create mode 100644 Media/Themes/Umami/Icon/apps/key_bindings.png create mode 100644 Media/Themes/Umami/Icon/apps/kfm.png create mode 100644 Media/Themes/Umami/Icon/apps/khelpcenter.png create mode 100644 Media/Themes/Umami/Icon/apps/konsole.png create mode 100644 Media/Themes/Umami/Icon/apps/krfb.png create mode 100644 Media/Themes/Umami/Icon/apps/kscreensaver.png create mode 100644 Media/Themes/Umami/Icon/apps/ksysguard.png create mode 100644 Media/Themes/Umami/Icon/apps/kuser.png create mode 100644 Media/Themes/Umami/Icon/apps/kwin.png create mode 100644 Media/Themes/Umami/Icon/apps/locale.png create mode 100644 Media/Themes/Umami/Icon/apps/mail_generic.png create mode 100644 Media/Themes/Umami/Icon/apps/office-calendar.png create mode 100644 Media/Themes/Umami/Icon/apps/openterm.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-accessibility.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-assistive-technology.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-font.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-keyboard-shortcuts.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-locale.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-multimedia.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-remote-desktop.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-screensaver.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-theme.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-desktop-wallpaper.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-system-network-proxy.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-system-session.png create mode 100644 Media/Themes/Umami/Icon/apps/preferences-system-windows.png create mode 100644 Media/Themes/Umami/Icon/apps/proxy-config.png create mode 100644 Media/Themes/Umami/Icon/apps/proxy.png create mode 100644 Media/Themes/Umami/Icon/apps/redhat-email.png create mode 100644 Media/Themes/Umami/Icon/apps/redhat-filemanager.png create mode 100644 Media/Themes/Umami/Icon/apps/redhat-web-browser.png create mode 100644 Media/Themes/Umami/Icon/apps/screensaver.png create mode 100644 Media/Themes/Umami/Icon/apps/stock_proxy.png create mode 100644 Media/Themes/Umami/Icon/apps/style.png create mode 100644 Media/Themes/Umami/Icon/apps/susehelpcenter.png create mode 100644 Media/Themes/Umami/Icon/apps/system-config-users.png create mode 100644 Media/Themes/Umami/Icon/apps/system-file-manager.png create mode 100644 Media/Themes/Umami/Icon/apps/system-installer.png create mode 100644 Media/Themes/Umami/Icon/apps/system-software-update.png create mode 100644 Media/Themes/Umami/Icon/apps/system-users.png create mode 100644 Media/Themes/Umami/Icon/apps/terminal.png create mode 100644 Media/Themes/Umami/Icon/apps/text-editor.png create mode 100644 Media/Themes/Umami/Icon/apps/update-manager.png create mode 100644 Media/Themes/Umami/Icon/apps/utilities-system-monitor.png create mode 100644 Media/Themes/Umami/Icon/apps/utilities-terminal.png create mode 100644 Media/Themes/Umami/Icon/apps/wallpaper.png create mode 100644 Media/Themes/Umami/Icon/apps/web-browser.png create mode 100644 Media/Themes/Umami/Icon/apps/xfcalendar.png create mode 100644 Media/Themes/Umami/Icon/apps/xfce-edit.png create mode 100644 Media/Themes/Umami/Icon/apps/xfce-filemanager.png create mode 100644 Media/Themes/Umami/Icon/apps/xfce-mail.png create mode 100644 Media/Themes/Umami/Icon/apps/xfce-man.png create mode 100644 Media/Themes/Umami/Icon/apps/xfce-terminal.png create mode 100644 Media/Themes/Umami/Icon/apps/xfce4-backdrop.png create mode 100644 Media/Themes/Umami/Icon/apps/xfce4-session.png create mode 100644 Media/Themes/Umami/Icon/apps/xfce4-ui.png create mode 100644 Media/Themes/Umami/Icon/apps/xfwm4.png create mode 100644 Media/Themes/Umami/Icon/apps/ximian-evolution-calendar.png create mode 100644 Media/Themes/Umami/Icon/apps/xscreensaver.png create mode 100644 Media/Themes/Umami/Icon/apps/zen-icon.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-accessories.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-development.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-games.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-graphics.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-internet.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-multimedia.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-office.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-other.png create mode 100644 Media/Themes/Umami/Icon/categories/applications-system.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-applications.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-control-center.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-devel.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-globe.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-graphics.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-joystick.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-multimedia.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-other.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-settings.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-system.png create mode 100644 Media/Themes/Umami/Icon/categories/gnome-util.png create mode 100644 Media/Themes/Umami/Icon/categories/gtk-preferences.png create mode 100644 Media/Themes/Umami/Icon/categories/input_devices_settings.png create mode 100644 Media/Themes/Umami/Icon/categories/kcontrol.png create mode 100644 Media/Themes/Umami/Icon/categories/package_development.png create mode 100644 Media/Themes/Umami/Icon/categories/package_games.png create mode 100644 Media/Themes/Umami/Icon/categories/package_graphics.png create mode 100644 Media/Themes/Umami/Icon/categories/package_multimedia.png create mode 100644 Media/Themes/Umami/Icon/categories/package_network.png create mode 100644 Media/Themes/Umami/Icon/categories/package_office.png create mode 100644 Media/Themes/Umami/Icon/categories/package_settings.png create mode 100644 Media/Themes/Umami/Icon/categories/package_system.png create mode 100644 Media/Themes/Umami/Icon/categories/package_utilities.png create mode 100644 Media/Themes/Umami/Icon/categories/preferences-desktop-peripherals.png create mode 100644 Media/Themes/Umami/Icon/categories/preferences-desktop.png create mode 100644 Media/Themes/Umami/Icon/categories/preferences-system.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-accessories.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-games.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-graphics.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-internet.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-office.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-preferences.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-programming.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-sound_video.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-system_settings.png create mode 100644 Media/Themes/Umami/Icon/categories/redhat-system_tools.png create mode 100644 Media/Themes/Umami/Icon/categories/stock_internet.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-devel.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-games.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-graphics.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-internet.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-multimedia.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-office.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-system-settings.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-system.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce-utils.png create mode 100644 Media/Themes/Umami/Icon/categories/xfce4-settings.png create mode 100644 Media/Themes/Umami/Icon/clipboard.png create mode 100644 Media/Themes/Umami/Icon/default.png create mode 100644 Media/Themes/Umami/Icon/devices/3floppy_unmount.png create mode 100644 Media/Themes/Umami/Icon/devices/audio-card.png create mode 100644 Media/Themes/Umami/Icon/devices/audio-input-microphone.png create mode 100644 Media/Themes/Umami/Icon/devices/battery.png create mode 100644 Media/Themes/Umami/Icon/devices/camera-photo.png create mode 100644 Media/Themes/Umami/Icon/devices/camera-video.png create mode 100644 Media/Themes/Umami/Icon/devices/camera.png create mode 100644 Media/Themes/Umami/Icon/devices/camera_unmount.png create mode 100644 Media/Themes/Umami/Icon/devices/cdrom_unmount.png create mode 100644 Media/Themes/Umami/Icon/devices/cdwriter_unmount.png create mode 100644 Media/Themes/Umami/Icon/devices/chardevice.png create mode 100644 Media/Themes/Umami/Icon/devices/computer.png create mode 100644 Media/Themes/Umami/Icon/devices/display.png create mode 100644 Media/Themes/Umami/Icon/devices/drive-cdrom.png create mode 100644 Media/Themes/Umami/Icon/devices/drive-harddisk.png create mode 100644 Media/Themes/Umami/Icon/devices/drive-optical.png create mode 100644 Media/Themes/Umami/Icon/devices/drive-removable-media.png create mode 100644 Media/Themes/Umami/Icon/devices/dvd_unmount.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-battery.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-cdrom-audio.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-cdrom.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-computer.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-disc-cdr.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-disc-cdrw.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdr-plus.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdr.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdram.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdrom.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdrw.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-dvd.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-ethernet.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-floppy.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-harddisk-1394.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-harddisk-usb.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-harddisk.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-ipod.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-keyboard.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-media-cf.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-media-ms.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-media-sdmmc.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-media-sm.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-mouse-ball.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-mouse-optical.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-printer.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-removable-1394.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-removable-usb.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-removable.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-dev-wavelan.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-fs-client.png create mode 100644 Media/Themes/Umami/Icon/devices/gnome-stock-mic.png create mode 100644 Media/Themes/Umami/Icon/devices/gtk-cdrom.png create mode 100644 Media/Themes/Umami/Icon/devices/gtk-floppy.png create mode 100644 Media/Themes/Umami/Icon/devices/gtk-harddisk.png create mode 100644 Media/Themes/Umami/Icon/devices/harddrive.png create mode 100644 Media/Themes/Umami/Icon/devices/hdd_unmount.png create mode 100644 Media/Themes/Umami/Icon/devices/input-gaming.png create mode 100644 Media/Themes/Umami/Icon/devices/input-keyboard.png create mode 100644 Media/Themes/Umami/Icon/devices/input-mouse.png create mode 100644 Media/Themes/Umami/Icon/devices/ipod_mount.png create mode 100644 Media/Themes/Umami/Icon/devices/joystick.png create mode 100644 Media/Themes/Umami/Icon/devices/keyboard.png create mode 100644 Media/Themes/Umami/Icon/devices/kjobviewer.png create mode 100644 Media/Themes/Umami/Icon/devices/kxkb.png create mode 100644 Media/Themes/Umami/Icon/devices/media-cdrom.png create mode 100644 Media/Themes/Umami/Icon/devices/media-flash.png create mode 100644 Media/Themes/Umami/Icon/devices/media-floppy.png create mode 100644 Media/Themes/Umami/Icon/devices/media-optical.png create mode 100644 Media/Themes/Umami/Icon/devices/mouse.png create mode 100644 Media/Themes/Umami/Icon/devices/multimedia-player.png create mode 100644 Media/Themes/Umami/Icon/devices/network-wired.png create mode 100644 Media/Themes/Umami/Icon/devices/network-wireless.png create mode 100644 Media/Themes/Umami/Icon/devices/nm-adhoc.png create mode 100644 Media/Themes/Umami/Icon/devices/nm-device-wired.png create mode 100644 Media/Themes/Umami/Icon/devices/nm-device-wireless.png create mode 100644 Media/Themes/Umami/Icon/devices/printer-remote.png create mode 100644 Media/Themes/Umami/Icon/devices/printer.png create mode 100644 Media/Themes/Umami/Icon/devices/printer1.png create mode 100644 Media/Themes/Umami/Icon/devices/printmgr.png create mode 100644 Media/Themes/Umami/Icon/devices/stock_mic.png create mode 100644 Media/Themes/Umami/Icon/devices/stock_printers.png create mode 100644 Media/Themes/Umami/Icon/devices/system-floppy.png create mode 100644 Media/Themes/Umami/Icon/devices/system.png create mode 100644 Media/Themes/Umami/Icon/devices/usbpendrive_unmount.png create mode 100644 Media/Themes/Umami/Icon/devices/video-display.png create mode 100644 Media/Themes/Umami/Icon/devices/xfce-printer.png create mode 100644 Media/Themes/Umami/Icon/devices/xfce4-display.png create mode 100644 Media/Themes/Umami/Icon/devices/xfce4-keyboard.png create mode 100644 Media/Themes/Umami/Icon/devices/xfce4-mouse.png create mode 100644 Media/Themes/Umami/Icon/devices/yast_HD.png create mode 100644 Media/Themes/Umami/Icon/devices/yast_idetude.png create mode 100644 Media/Themes/Umami/Icon/devices/yast_joystick.png create mode 100644 Media/Themes/Umami/Icon/devices/yast_mouse.png create mode 100644 Media/Themes/Umami/Icon/devices/yast_printer.png create mode 100644 Media/Themes/Umami/Icon/devices/yast_soundcard.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-favorite.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-important.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-noread.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-nowrite.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-photos.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-readonly.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-symbolic-link.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-system.png create mode 100644 Media/Themes/Umami/Icon/emblems/emblem-unreadable.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-angel.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-crying.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-devilish.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-glasses.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-grin.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-kiss.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-monkey.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-plain.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-sad.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-smile-big.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-smile.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-surprise.png create mode 100644 Media/Themes/Umami/Icon/emotes/face-wink.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-1.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-11.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-13.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-18.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-2.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-22.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-3.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-4.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-5.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-6.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-7.png create mode 100644 Media/Themes/Umami/Icon/emotes/stock_smiley-8.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-certificate.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.ms-excel.sheet.macroEnabled.12.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.ms-powerpoint.presentation.macroEnabled.12.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.ms-word.document.macroEnabled.12.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.openxmlformats-officedocument.presentationml.presentation.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.openxmlformats-officedocument.presentationml.template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.openxmlformats-officedocument.spreadsheetml.sheet.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.openxmlformats-officedocument.spreadsheetml.template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.openxmlformats-officedocument.wordprocessingml.document.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-vnd.openxmlformats-officedocument.wordprocessingml.template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/application-x-executable.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/ascii.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/audio-x-generic.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/binary.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/contents2.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/deb.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/document.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/empty.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/exec.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/folder_tar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/font-x-generic.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/font.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/font_bitmap.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/font_truetype.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/font_type1.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-fs-executable.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-magicpoint.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-msword.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-ogg.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-pdf.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-postscript.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-rtf.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.lotus-1-2-3.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.ms-excel.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.ms-powerpoint.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.graphics-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.graphics.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.image.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.presentation-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.presentation.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.spreadsheet-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.spreadsheet.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.text-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.text-web.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.text.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.rn-realmedia-secure.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.rn-realmedia-vbr.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.rn-realmedia.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.stardivision.calc.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.stardivision.impress.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.stardivision.writer.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.calc.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.calc.template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.draw.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.draw.template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.impress.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.impress.template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.writer.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.writer.template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-wordperfect.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-7z-compressed.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-abiword.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-applix-spreadsheet.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-applix-word.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-archive.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-arj.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-bzip-compressed-tar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-bzip.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-compress.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-compressed-tar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-cpio-compressed.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-cpio.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-deb.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-dvi.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-executable.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-afm.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-bdf.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-linux-psf.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-pcf.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-sunos-news.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-ttf.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-gnumeric.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-gzip.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-gzpostscript.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-jar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-killustrator.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-kpresenter.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-kspread.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-kword.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lha.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lhz.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lzma-compressed-tar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lzma.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-ms-dos-executable.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-perl.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-php.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-python-bytecode.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-rar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-rpm.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-scribus.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-shellscript.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-shockwave-flash.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-stuffit.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tarz.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tex.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-xhtml+xml.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-zip.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-audio.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-image.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-html.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-vnd.wap.wml.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-x-csh.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-x-python.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-x-sh.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-x-vcalendar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-x-vcard.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-x-zsh.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-text.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-video.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-mime-x-font-afm.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/gnome-package.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/html.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/image-x-generic.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/image.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/kpresenter_kpr.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/mime_ascii.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/misc.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/package-x-generic.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/package.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/package_editors.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/package_wordprocessing.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/plan.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/rpm.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/shellscript.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/sound.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/spreadsheet.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/stock_addressbook.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/stock_calendar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/stock_certificate.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/stock_script.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/tar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/template_source.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/text-html.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/text-x-generic-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/text-x-generic.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/text-x-script.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/tgz.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/txt.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/txt2.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/unknown.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/vcalendar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/vcard.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/video-x-generic.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/video.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/wordprocessing.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/www.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-address-book.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-calendar.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-document-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-document.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-drawing-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-drawing.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-presentation-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-presentation.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-spreadsheet-template.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/x-office-spreadsheet.png create mode 100644 Media/Themes/Umami/Icon/mimetypes/zip.png create mode 100644 Media/Themes/Umami/Icon/places/application-x-gnome-saved-search.png create mode 100644 Media/Themes/Umami/Icon/places/desktop.png create mode 100644 Media/Themes/Umami/Icon/places/distributor-logo.png create mode 100644 Media/Themes/Umami/Icon/places/emptytrash.png create mode 100644 Media/Themes/Umami/Icon/places/folder-remote.png create mode 100644 Media/Themes/Umami/Icon/places/folder-saved-search.png create mode 100644 Media/Themes/Umami/Icon/places/folder.png create mode 100644 Media/Themes/Umami/Icon/places/folder_home.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-desktop.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-directory.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-ftp.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-home.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-network.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-nfs.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-server.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-share.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-smb.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-ssh.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-fs-trash-empty.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-main-menu.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-mime-x-directory-nfs-server.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-mime-x-directory-smb-server.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-mime-x-directory-smb-share.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-mime-x-directory-smb-workgroup.png create mode 100644 Media/Themes/Umami/Icon/places/gnome-stock-trash.png create mode 100644 Media/Themes/Umami/Icon/places/gtk-directory.png create mode 100644 Media/Themes/Umami/Icon/places/gtk-network.png create mode 100644 Media/Themes/Umami/Icon/places/inode-directory.png create mode 100644 Media/Themes/Umami/Icon/places/network-server.png create mode 100644 Media/Themes/Umami/Icon/places/network-workgroup.png create mode 100644 Media/Themes/Umami/Icon/places/network.png create mode 100644 Media/Themes/Umami/Icon/places/network_local.png create mode 100644 Media/Themes/Umami/Icon/places/novell-button.png create mode 100644 Media/Themes/Umami/Icon/places/redhat-network-server.png create mode 100644 Media/Themes/Umami/Icon/places/server.png create mode 100644 Media/Themes/Umami/Icon/places/start-here.png create mode 100644 Media/Themes/Umami/Icon/places/stock_folder.png create mode 100644 Media/Themes/Umami/Icon/places/trashcan_empty.png create mode 100644 Media/Themes/Umami/Icon/places/user-desktop.png create mode 100644 Media/Themes/Umami/Icon/places/user-home.png create mode 100644 Media/Themes/Umami/Icon/places/user-trash.png create mode 100644 Media/Themes/Umami/Icon/places/xfce-trash_empty.png create mode 100644 Media/Themes/Umami/Icon/status/audio-volume-high.png create mode 100644 Media/Themes/Umami/Icon/status/audio-volume-low.png create mode 100644 Media/Themes/Umami/Icon/status/audio-volume-medium.png create mode 100644 Media/Themes/Umami/Icon/status/audio-volume-muted.png create mode 100644 Media/Themes/Umami/Icon/status/battery-caution.png create mode 100644 Media/Themes/Umami/Icon/status/connect_creating.png create mode 100644 Media/Themes/Umami/Icon/status/connect_established.png create mode 100644 Media/Themes/Umami/Icon/status/connect_no.png create mode 100644 Media/Themes/Umami/Icon/status/dialog-error.png create mode 100644 Media/Themes/Umami/Icon/status/dialog-information.png create mode 100644 Media/Themes/Umami/Icon/status/dialog-warning.png create mode 100644 Media/Themes/Umami/Icon/status/edittrash.png create mode 100644 Media/Themes/Umami/Icon/status/error.png create mode 100644 Media/Themes/Umami/Icon/status/folder-drag-accept.png create mode 100644 Media/Themes/Umami/Icon/status/folder-open.png create mode 100644 Media/Themes/Umami/Icon/status/folder-visiting.png create mode 100644 Media/Themes/Umami/Icon/status/folder_open.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-dev-wavelan-encrypted.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-fs-directory-accept.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-fs-directory-visiting.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-fs-loading-icon.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-fs-trash-full.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-netstatus-disconn.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-netstatus-error.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-netstatus-idle.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-netstatus-rx.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-netstatus-tx.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-netstatus-txrx.png create mode 100644 Media/Themes/Umami/Icon/status/gnome-stock-trash-full.png create mode 100644 Media/Themes/Umami/Icon/status/gtk-dialog-error.png create mode 100644 Media/Themes/Umami/Icon/status/gtk-dialog-info.png create mode 100644 Media/Themes/Umami/Icon/status/gtk-dialog-warning.png create mode 100644 Media/Themes/Umami/Icon/status/gtk-directory.png create mode 100644 Media/Themes/Umami/Icon/status/gtk-missing-image.png create mode 100644 Media/Themes/Umami/Icon/status/image-loading.png create mode 100644 Media/Themes/Umami/Icon/status/image-missing.png create mode 100644 Media/Themes/Umami/Icon/status/important.png create mode 100644 Media/Themes/Umami/Icon/status/info.png create mode 100644 Media/Themes/Umami/Icon/status/mail-attachment.png create mode 100644 Media/Themes/Umami/Icon/status/messagebox_critical.png create mode 100644 Media/Themes/Umami/Icon/status/messagebox_critical_16x16.png create mode 100644 Media/Themes/Umami/Icon/status/messagebox_info.png create mode 100644 Media/Themes/Umami/Icon/status/messagebox_info_16x16.png create mode 100644 Media/Themes/Umami/Icon/status/messagebox_warning.png create mode 100644 Media/Themes/Umami/Icon/status/messagebox_warning_16x16.png create mode 100644 Media/Themes/Umami/Icon/status/network-error.png create mode 100644 Media/Themes/Umami/Icon/status/network-idle.png create mode 100644 Media/Themes/Umami/Icon/status/network-offline.png create mode 100644 Media/Themes/Umami/Icon/status/network-receive.png create mode 100644 Media/Themes/Umami/Icon/status/network-transmit-receive.png create mode 100644 Media/Themes/Umami/Icon/status/network-transmit.png create mode 100644 Media/Themes/Umami/Icon/status/network-wireless-encrypted.png create mode 100644 Media/Themes/Umami/Icon/status/nm-no-connection.png create mode 100644 Media/Themes/Umami/Icon/status/printer-error.png create mode 100644 Media/Themes/Umami/Icon/status/software-update-available.png create mode 100644 Media/Themes/Umami/Icon/status/software-update-urgent.png create mode 100644 Media/Themes/Umami/Icon/status/stock_attach.png create mode 100644 Media/Themes/Umami/Icon/status/stock_dialog-error.png create mode 100644 Media/Themes/Umami/Icon/status/stock_dialog-info.png create mode 100644 Media/Themes/Umami/Icon/status/stock_dialog-warning.png create mode 100644 Media/Themes/Umami/Icon/status/stock_open.png create mode 100644 Media/Themes/Umami/Icon/status/stock_trash_full.png create mode 100644 Media/Themes/Umami/Icon/status/stock_volume-0.png create mode 100644 Media/Themes/Umami/Icon/status/stock_volume-max.png create mode 100644 Media/Themes/Umami/Icon/status/stock_volume-med.png create mode 100644 Media/Themes/Umami/Icon/status/stock_volume-min.png create mode 100644 Media/Themes/Umami/Icon/status/stock_volume-mute.png create mode 100644 Media/Themes/Umami/Icon/status/stock_volume.png create mode 100644 Media/Themes/Umami/Icon/status/stock_weather-cloudy.png create mode 100644 Media/Themes/Umami/Icon/status/stock_weather-few-clouds.png create mode 100644 Media/Themes/Umami/Icon/status/stock_weather-night-clear.png create mode 100644 Media/Themes/Umami/Icon/status/stock_weather-night-few-clouds.png create mode 100644 Media/Themes/Umami/Icon/status/stock_weather-showers.png create mode 100644 Media/Themes/Umami/Icon/status/stock_weather-snow.png create mode 100644 Media/Themes/Umami/Icon/status/stock_weather-storm.png create mode 100644 Media/Themes/Umami/Icon/status/stock_weather-sunny.png create mode 100644 Media/Themes/Umami/Icon/status/sunny.png create mode 100644 Media/Themes/Umami/Icon/status/trashcan_full.png create mode 100644 Media/Themes/Umami/Icon/status/user-trash-full.png create mode 100644 Media/Themes/Umami/Icon/status/weather-clear-night.png create mode 100644 Media/Themes/Umami/Icon/status/weather-clear.png create mode 100644 Media/Themes/Umami/Icon/status/weather-few-clouds-night.png create mode 100644 Media/Themes/Umami/Icon/status/weather-few-clouds.png create mode 100644 Media/Themes/Umami/Icon/status/weather-overcast.png create mode 100644 Media/Themes/Umami/Icon/status/weather-severe-alert.png create mode 100644 Media/Themes/Umami/Icon/status/weather-showers-scattered.png create mode 100644 Media/Themes/Umami/Icon/status/weather-showers.png create mode 100644 Media/Themes/Umami/Icon/status/weather-snow.png create mode 100644 Media/Themes/Umami/Icon/status/weather-storm.png create mode 100644 Media/Themes/Umami/Icon/status/xfce-trash_full.png create mode 100644 Media/Themes/Umami/Pointer/alternate.png create mode 100644 Media/Themes/Umami/Pointer/cross.png create mode 100644 Media/Themes/Umami/Pointer/dgn1.png create mode 100644 Media/Themes/Umami/Pointer/dgn2.png create mode 100644 Media/Themes/Umami/Pointer/help.png create mode 100644 Media/Themes/Umami/Pointer/horz.png create mode 100644 Media/Themes/Umami/Pointer/link.png create mode 100644 Media/Themes/Umami/Pointer/move.png create mode 100644 Media/Themes/Umami/Pointer/pen.png create mode 100644 Media/Themes/Umami/Pointer/pointer.png create mode 100644 Media/Themes/Umami/Pointer/text.png create mode 100644 Media/Themes/Umami/Pointer/unavailable.png create mode 100644 Media/Themes/Umami/Pointer/vert.png create mode 100644 Media/Themes/Umami/Pointer/wait.png create mode 100644 Media/Themes/Umami/Theme.HC create mode 100644 Media/Themes/Umami/Widget/checkbox.png create mode 100644 Media/Themes/Umami/Widget/checkbox_active.png create mode 100644 Media/Themes/Umami/Widget/checkbox_checked.png create mode 100644 Media/Themes/Umami/Widget/checkbox_checked_active.png create mode 100644 Media/Themes/Umami/Widget/horz_slider.png create mode 100644 Media/Themes/Umami/Widget/radio.png create mode 100644 Media/Themes/Umami/Widget/radio_active.png create mode 100644 Media/Themes/Umami/Widget/radio_selected.png create mode 100644 Media/Themes/Umami/Widget/sb_down_button.png create mode 100644 Media/Themes/Umami/Widget/sb_left_button.png create mode 100644 Media/Themes/Umami/Widget/sb_right_button.png create mode 100644 Media/Themes/Umami/Widget/sb_up_button.png create mode 100644 Media/Themes/Umami/Widget/vert_slider.png create mode 100644 Media/Themes/Umami/Window/corner_resize.png create mode 100644 Media/Themes/Umami/Window/min_max_close.png create mode 100644 Media/Themes/Umami/Window/min_max_close_pressed.png create mode 100644 Media/Themes/Umami/wallpaper.jpg create mode 100644 Media/Themes/Umami/wallpaper_old.png create mode 100644 Run.HC create mode 100644 Settings/SystemMenu.json create mode 100644 System/Api/Dns.HC create mode 100644 System/Api/Icmp.HC create mode 100644 System/Api/Ipv4.HC create mode 100644 System/Api/MD5.HC create mode 100644 System/Api/NetInfo.HC create mode 100644 System/Api/Tcp.HC create mode 100644 System/Api/Tls.HC create mode 100644 System/Config/Net.json create mode 100644 System/Core/Compositor.HC create mode 100644 System/Core/FileSystem.HC create mode 100644 System/Core/Menu.HC create mode 100644 System/Core/MessageBox.HC create mode 100644 System/Core/Scheduler.HC create mode 100644 System/Core/Shell.HC create mode 100644 System/Core/ShellCommands.HC create mode 100644 System/Core/SystemStarter.HC create mode 100644 System/Core/SystemTray.HC create mode 100644 System/Drivers/AC97.HC create mode 100644 System/Drivers/Audio.HC create mode 100644 System/Drivers/Display.HC create mode 100644 System/Drivers/Mouse.HC create mode 100644 System/Drivers/Pci.HC create mode 100644 System/Drivers/VMSVGA.HC create mode 100644 System/Drivers/VMwareTools.HC create mode 100644 System/Drivers/Virtio-blk.HC create mode 100644 System/FFI/Base.HC create mode 100644 System/FFI/ELF64.HC create mode 100644 System/FFI/LibC.HC create mode 100644 System/FFI/New.HC create mode 100644 System/Jakt/DC.HC create mode 100644 System/Jakt/IOPort.HC create mode 100644 System/Jakt/Input.HC create mode 100644 System/Jakt/OS.HC create mode 100644 System/Jakt/PCI.HC create mode 100644 System/Jakt/Time.HC create mode 100644 System/Jakt/Window.HC create mode 100644 System/Libraries/Animation2D.HC create mode 100644 System/Libraries/Audio.HC create mode 100644 System/Libraries/Base64.HC create mode 100644 System/Libraries/BitmapFont.HC create mode 100644 System/Libraries/Clipboard.HC create mode 100644 System/Libraries/Display.HC create mode 100644 System/Libraries/FileSystem.HC create mode 100644 System/Libraries/Function.HC create mode 100644 System/Libraries/Graphics2D.HC create mode 100644 System/Libraries/Gui.HC create mode 100644 System/Libraries/Http.HC create mode 100644 System/Libraries/Image.HC create mode 100644 System/Libraries/Ipc.HC create mode 100644 System/Libraries/Json.HC create mode 100644 System/Libraries/RawText.HC create mode 100644 System/Libraries/Rsa.HC create mode 100644 System/Libraries/Session.HC create mode 100644 System/Libraries/Sha256.HC create mode 100644 System/Libraries/Shell.HC create mode 100644 System/Libraries/Stdio.HC create mode 100644 System/Libraries/String.HC create mode 100644 System/Libraries/System.HC create mode 100644 System/Libraries/Theme.HC create mode 100644 System/Libraries/Tlse.HC create mode 100644 System/Libraries/User.HC create mode 100644 System/Libraries/Widget.HC create mode 100644 System/MakeSystem.HC create mode 100644 System/Setup/Environment.HC create mode 100644 System/Setup/Init.HC create mode 100644 System/Shell/Commands/aplay.HC create mode 100644 System/Shell/Commands/cat.HC create mode 100644 System/Shell/Commands/clear.HC create mode 100644 System/Shell/Commands/echo.HC create mode 100644 System/Shell/Commands/esh.HC create mode 100644 System/Shell/Commands/history.HC create mode 100644 System/Shell/Commands/ifconfig.HC create mode 100644 System/Shell/Commands/ls.HC create mode 100644 System/Shell/Commands/nslookup.HC create mode 100644 System/Shell/Commands/open.HC create mode 100644 System/Shell/Commands/ping.HC create mode 100644 System/Shell/Commands/poweroff.HC create mode 100644 System/Shell/Commands/pwd.HC create mode 100644 System/Shell/Commands/reboot.HC create mode 100644 System/Shell/Commands/sh.HC create mode 100644 System/Shell/Commands/uname.HC create mode 100644 System/Shell/Commands/whoami.HC create mode 100644 System/Shell/Commands/wpset.HC create mode 100644 System/Utilities/Dns.HC create mode 100644 System/Utilities/Image.HC create mode 100644 System/Utilities/NetRep.HC create mode 100644 System/Utilities/Ping.HC create mode 100644 System/Utilities/Time.HC create mode 100755 scripts/build-all create mode 100644 src/image/image.c create mode 100644 src/image/stb_image.h create mode 100644 src/image/stb_image_write.h create mode 100644 src/libtemple/ioport.h create mode 100644 src/libtemple/libtemple.cpp create mode 100644 src/libtemple/os.h create mode 100644 src/libtemple/pci.h create mode 100644 src/libtemple/time.h create mode 100644 src/net/devices/virtio.h create mode 100644 src/net/devices/virtio.jakt create mode 100644 src/net/lib/json.jakt create mode 100644 src/net/lib/util.jakt create mode 100644 src/net/net.jakt create mode 100644 src/net/os/ioport.jakt create mode 100644 src/net/os/os.jakt create mode 100644 src/net/os/pci.jakt create mode 100644 src/net/os/time.jakt create mode 100644 src/net/tcpip.jakt create mode 100644 src/tlse/LICENSE create mode 100644 src/tlse/libtomcrypt.c create mode 100644 src/tlse/tlse.c create mode 100644 src/tlse/tlse.h diff --git a/.clang-format b/.clang-format new file mode 100644 index 0000000..61e0467 --- /dev/null +++ b/.clang-format @@ -0,0 +1,14 @@ +--- +Language: Cpp +BasedOnStyle: WebKit +SpaceAfterTemplateKeyword: false +AlignEscapedNewlines: Left +AlignTrailingComments: true +BreakBeforeInheritanceComma: true +BreakConstructorInitializers: BeforeComma +IndentPPDirectives: AfterHash +BreakBeforeBraces: Custom +BraceWrapping: + AfterFunction: true +NamespaceIndentation: None +QualifierAlignment: Right diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..f9613a5 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,2 @@ +src/image/** linguist-vendored +src/tlse/** linguist-vendored diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..0de6233 --- /dev/null +++ b/.gitignore @@ -0,0 +1,3 @@ +build/ +reports/ +src/net/exports/ diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..fb63315 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,10 @@ +{ + "jaktLanguageServer.compiler.executablePath": "jakt", + "git.ignoreLimitWarning": true, + "pmd-cpd.language": [ + "cpp" + ], + "pmd-cpd.onStartBehavior": "Show", + "pmd-cpd.sourceDirectory": "/home/alec/repos/erythros", + "terminal.integrated.shellIntegration.history": 0 +} diff --git a/.vscode/tasks.json b/.vscode/tasks.json new file mode 100644 index 0000000..370f88e --- /dev/null +++ b/.vscode/tasks.json @@ -0,0 +1 @@ +{"version":"2.0.0","tasks":[{"label":"Build All","type":"shell","command":"${workspaceFolder}/scripts/build-all ${workspaceFolder}","group":{"kind":"build","isDefault":true},"problemMatcher":[]}]} diff --git a/Applets/Calendar.applet/Run.HC b/Applets/Calendar.applet/Run.HC new file mode 100644 index 0000000..e9048fb --- /dev/null +++ b/Applets/Calendar.applet/Run.HC @@ -0,0 +1,5 @@ +Gui.App(); + +U0 Main() { Suspend; } + +Main; diff --git a/Applets/ClipManager.applet/Run.HC b/Applets/ClipManager.applet/Run.HC new file mode 100644 index 0000000..9854994 --- /dev/null +++ b/Applets/ClipManager.applet/Run.HC @@ -0,0 +1,32 @@ +Gui.App(); + +Context2DWidget* clipmanager_icon = SystemTray.RegisterItem(); + +Context2D* ctx_clip_icon = Image.FileToContext2D("M:/Media/Themes/Umami/Icon/clipboard.png"); + +CopyRect2D(clipmanager_icon->ctx, 0, 0, ctx_clip_icon); + +U0 Main() +{ + I64 length = 0; + I64 text_width = 0; + while (1) { + if (length < Clipboard.Length()) { + CopyRect2D(clipmanager_icon->ctx, 0, 0, ctx_clip_icon); + Rect2D(clipmanager_icon->ctx, 12, 12, 12, 12, Color(255, 0, 0)); + text_width = Print2D(NULL, Compositor.theme.font.sans, 0, 0, + Color(255, 255, 255), , "%02d", Clipboard.Length()); + Print2D(clipmanager_icon->ctx, Compositor.theme.font.sans, + 11 + 6 - (text_width / 2), 13, Color(255, 255, 255), , "%02d", + Clipboard.Length()); + Print2D(clipmanager_icon->ctx, Compositor.theme.font.sans, + 11 + 7 - (text_width / 2), 13, Color(255, 255, 255), , "%02d", + Clipboard.Length()); + length = Clipboard.Length(); + Gui.Window.Refresh(Compositor.menubar.win); + } + Sleep(1); + } +} + +Main; diff --git a/Applets/NetworkStatus.applet/Run.HC b/Applets/NetworkStatus.applet/Run.HC new file mode 100644 index 0000000..ef3d13e --- /dev/null +++ b/Applets/NetworkStatus.applet/Run.HC @@ -0,0 +1,101 @@ +Gui.App(); + +U64 flags = WIN_FLAGS_MOVABLE | WIN_FLAGS_ICON | WIN_FLAGS_TITLE_BAR | WIN_FLAGS_CLOSE_BUTTON | WIN_FLAGS_SKIP; +Window* win = Compositor.CreateWindow(Display.Width() - 240, 33, 240, 120, flags); +Gui.Window.SetIcon(win, Image.FileToContext2D("window_icon_16x16.png")); +Gui.Window.Hide(win); +Gui.Window.SetTitle(win, "Network Status"); + +TextLabelWidget* status_label1 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 16, 16, 192, 16); +TextLabelWidget* status_label2 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 16, 32, 192, 16); +TextLabelWidget* status_label3 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 16, 48, 192, 16); +TextLabelWidget* status_label4 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 16, 64, 192, 16); + +TextLabelWidget* status_label5 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 128, 16, 192, 16); +TextLabelWidget* status_label6 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 128, 32, 192, 16); +TextLabelWidget* status_label7 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 128, 48, 192, 16); +TextLabelWidget* status_label8 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 128, 64, 192, 16); +Gui.Widget.SetFont(status_label1, "Eight Bit Dragon"); +Gui.Widget.SetFont(status_label2, "Eight Bit Dragon"); +Gui.Widget.SetFont(status_label3, "Eight Bit Dragon"); +Gui.Widget.SetFont(status_label4, "Eight Bit Dragon"); +Gui.Widget.SetFont(status_label5, "Eight Bit Dragon"); +Gui.Widget.SetFont(status_label6, "Eight Bit Dragon"); +Gui.Widget.SetFont(status_label7, "Eight Bit Dragon"); +Gui.Widget.SetFont(status_label8, "Eight Bit Dragon"); + +Context2DWidget* network_icon = SystemTray.RegisterItem(); + +Context2D* ctx_network_error = Image.FileToContext2D("M:/Media/Themes/Umami/Icon/status/network-error.png"); +Context2D* ctx_network_idle = Image.FileToContext2D("M:/Media/Themes/Umami/Icon/status/network-idle.png"); +Context2D* ctx_network_offline = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/network-offline.png"); +Context2D* ctx_network_rx = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/network-receive.png"); +Context2D* ctx_network_tx = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/network-transmit.png"); +Context2D* ctx_network_txrx = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/network-transmit-receive.png"); +Context2D* ctx_network_wireless_enc = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/network-wireless-encrypted.png"); + +U0 @networkstatus_show(Widget*) +{ + + NetInfoRequest* req = @net_info_request; + + win->flags &= ~WIN_FLAGS_SKIP; + + U8 status_ipaddr[128]; + U8 status_netmask[128]; + U8 status_gateway[128]; + U8 status_dns1[128]; + + StrPrint(&status_ipaddr, "%d.%d.%d.%d", req->ipv4_address.u8[3], req->ipv4_address.u8[2], + req->ipv4_address.u8[1], req->ipv4_address.u8[0]); + StrPrint(&status_netmask, "%d.%d.%d.%d", req->ipv4_netmask.u8[3], req->ipv4_netmask.u8[2], + req->ipv4_netmask.u8[1], req->ipv4_netmask.u8[0]); + StrPrint(&status_gateway, "%d.%d.%d.%d", req->ipv4_gateway.u8[3], req->ipv4_gateway.u8[2], + req->ipv4_gateway.u8[1], req->ipv4_gateway.u8[0]); + StrPrint(&status_dns1, "%d.%d.%d.%d", req->dns_server_address.u8[3], req->dns_server_address.u8[2], + req->dns_server_address.u8[1], req->dns_server_address.u8[0], req->dns_server_port); + + Gui.Widget.SetText(status_label1, "IP address:"); + Gui.Widget.SetText(status_label2, "Subnet mask:"); + Gui.Widget.SetText(status_label3, "Gateway:"); + Gui.Widget.SetText(status_label4, "DNS Server:"); + + Gui.Widget.SetText(status_label5, &status_ipaddr); + Gui.Widget.SetText(status_label6, &status_netmask); + Gui.Widget.SetText(status_label7, &status_gateway); + Gui.Widget.SetText(status_label8, &status_dns1); + + Gui.Window.SetFocus(win); + Gui.Window.Refresh(win); + while (!Gui.Window.IsVisible(win)) + Compositor.ShowWindow(win); +} + +U0 @networkstatus_hide(Window*) +{ + win->flags |= WIN_FLAGS_SKIP; + Compositor.HideWindow(win); +} + +network_icon->ctx = ctx_network_error; +Gui.Window.Refresh(Compositor.menubar.win); +Gui.Window.SetCallback(win, "close", &@networkstatus_hide); + +U0 Main() +{ + Gui.Widget.SetCallback(network_icon, "clicked", &@networkstatus_show); + while (1) { + if (network_icon->ctx == ctx_network_error) { + network_icon->ctx = ctx_network_idle; + Gui.Window.Refresh(Compositor.menubar.win); + } + Sleep(10); + } +} + +Main; diff --git a/Applets/NetworkStatus.applet/window_icon_16x16.png b/Applets/NetworkStatus.applet/window_icon_16x16.png new file mode 100644 index 0000000000000000000000000000000000000000..12d636979083c744c5ab6b285f7056793cc03085 GIT binary patch literal 613 zcmV-r0-F7aP)004R> z004l5008;`004mK004C_008P=0026d000+od=W=3o*(4nlr~DdpB7Kg!oo32hN%InK@%rl~Q*mZ{N-He8PLN3ZROrvXcknthJ9U z+ms~9#OTPV?d$8O8dr(q7;8h)G$BnBk~AgFHhr_%v~zQf@s0sSQP?<+89F_N0t!@2 zfOx?KhBypSU_HrLn7KjW3fnP2#35b?tflAJN%qwu4)xUd`uz`sM|-Is?8Z6I=MQf& z#vo3x+qvUvmFejxjGj7x@~zE%e6V!~YXQqe9VbE%1UPZnJ#sTU`w}Ozm0+*jPIA0g zz#!tV|HHlTe2k5abws?g&Iuxds>gXxXe}$v4fc1(n4rRsUw_G+>)?L);6BC}7e&$I z%>?355BC$S|%Ob!^0;qCdgIfT|0qS&*GO)Y%Qj?y(%5tXV0Bs zWqGCc`qi6x>s8$9>Z<+t^g2-#5rq+<4KYE`Y7~khC(rYen|qT}0NBv3n~`f*r?M;? zS5;g73vIoB=}VQ|GtXY!DrjOFx}+hj;BRPYmsEm-%|5(GY-t))9A4NB*6t00000NkvXXu0mjfj=vQn literal 0 HcmV?d00001 diff --git a/Applets/VolumeSlider.applet/Run.HC b/Applets/VolumeSlider.applet/Run.HC new file mode 100644 index 0000000..6af2e3f --- /dev/null +++ b/Applets/VolumeSlider.applet/Run.HC @@ -0,0 +1,71 @@ +Gui.App(); + +Window* win = Compositor.CreateWindow(Display.Width(), 32, 256, 32, + WIN_FLAGS_NOHILIGHT | WIN_FLAGS_SKIP); + +Context2DWidget* volume_slider_icon = Gui.CreateWidget(win, WIDGET_TYPE_CONTEXT2D, 0, 0, 24, 24); +volume_slider_icon->ctx = NewContext2D(24, 24); +HorizontalSliderWidget* volume_slider = Gui.CreateWidget(win, WIDGET_TYPE_HORZ_SLIDER, 32, 4, 212, 24); +volume_slider->max = 100; +volume_slider->scroll = volume_slider->width; + +Context2DWidget* tray_icon = SystemTray.RegisterItem(); + +Context2D* ctx_volume_muted = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/audio-volume-muted.png"); +Context2D* ctx_volume_low = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/audio-volume-low.png"); +Context2D* ctx_volume_medium = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/audio-volume-medium.png"); +Context2D* ctx_volume_high = Image.FileToContext2D( + "M:/Media/Themes/Umami/Icon/status/audio-volume-high.png"); + +U0 @volumeslider_show(Widget*) +{ + win->x = Display.Width() - 256; + Compositor.ShowWindow(win); + Gui.Window.SetFocus(win); +} + +U0 @volumeslider_change(Widget*) +{ + I64 volume = Min(ToI64(volume_slider->scroll * 0.47), 100); + Audio.mixer.left = volume; + Audio.mixer.right = volume; + Context2D* update_slider_icon = NULL; + switch (volume) { + case 0: + update_slider_icon = ctx_volume_muted; + break; + case 1...24: + update_slider_icon = ctx_volume_low; + break; + case 25...74: + update_slider_icon = ctx_volume_medium; + break; + case 75...100: + default: + update_slider_icon = ctx_volume_high; + break; + } + CopyRect2D(volume_slider_icon->ctx, 0, 0, update_slider_icon); + CopyRect2D(tray_icon->ctx, 0, 0, update_slider_icon); + Gui.Window.Refresh(Compositor.menubar.win); +} +@volumeslider_change(NULL); + +U0 Main() +{ + Gui.Window.SetTitle(win, "VolumeSlider"); + Compositor.HideWindow(win); + Gui.Widget.SetCallback(tray_icon, "clicked", &@volumeslider_show); + Gui.Widget.SetCallback(volume_slider, "change", &@volumeslider_change); + + while (1) { + if (Gui.Window.IsVisible(win) && Compositor.active_win != win) + Compositor.HideWindow(win); + Sleep(1); + } +} + +Main; \ No newline at end of file diff --git a/Applications/Accessories/Calculator.app/Run.HC b/Applications/Accessories/Calculator.app/Run.HC new file mode 100644 index 0000000..2de14ec --- /dev/null +++ b/Applications/Accessories/Calculator.app/Run.HC @@ -0,0 +1,100 @@ +Gui.App(); + +#define CALC_OP_ADD 0 +#define CALC_OP_SUB 1 +#define CALC_OP_MUL 2 +#define CALC_OP_DIV 3 + +class @calc_entry +{ + I64 op; + F64 entry; +}; + +Window* win = NULL; +TextInputWidget* result = NULL; + +U8* calc_font = "Eight Bit Dragon"; +U8* @calc_button_label[27] = { "Back", "CE", "C", "MC", "7", "8", "9", + "/", "sqrt", "MR", "4", "5", "6", "*", + "%", "MS", "1", "2", "3", "-", "1/x", + "M+", "0", "+/-", ".", "+", "=" }; + +U0 @calc_keypress_callback(Window*, I64 key) +{ + if (win != Compositor.active_win) + return; + U8* res_text = result->text; + switch (ScanCode2Char(key)) { + case '0' ... '9': + StrPrint(res_text + StrLen(res_text), "%c", ScanCode2Char(key)); + break; + } +} + +U0 @calc_btn_callback(ButtonWidget* widget) +{ + U8* res_text = result->text; + if (Str2I64(widget->text)) { + StrCpy(res_text + StrLen(res_text), widget->text); + } + if (!StrCmp(widget->text, "0")) + StrCpy(res_text + StrLen(res_text), widget->text); +} + +U0 window_close(Window* win) +{ + if (win == Compositor.active_win) + Gui.Window.SetFocus(Compositor.GetWindowByTitle("Wallpaper")); + Compositor.UnregisterForGlobalInputEvents(win); + Compositor.DestroyWindow(win); +} + +U0 Main() +{ + I64 i; + I64 j; + I64 k; + U64 flags = WIN_FLAGS_DEFAULT; + flags &= ~WIN_FLAGS_RESIZABLE; + flags &= ~WIN_FLAGS_MAX_BUTTON; + win = Compositor.CreateWindow(224, 224, 233, 240, flags); + Gui.Window.SetTitle(win, "Calculator"); + Compositor.RegisterForGlobalInputEvents(win); + Gui.Window.SetIcon(win, Image.FileToContext2D("window_icon_16x16.png")); + Gui.Window.SetFocus(win); + Gui.Window.SetCallback(win, "close", &window_close); + Gui.Window.SetCallback(win, "keypress", &@calc_keypress_callback); + + result = Gui.CreateWidget(win, WIDGET_TYPE_INPUT, 4, 4, 246, 24); + Gui.Widget.SetFont(result, calc_font); + + ButtonWidget* calc_btn[27]; + + for (i = 0; i < 4; i++) { + if (i < 3) { + calc_btn[i] = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, 46 + (60 * i), 32, 56, 32); + calc_btn[i]->color = Color(255, 0, 0); + Gui.Widget.SetText(calc_btn[i], @calc_button_label[i]); + Gui.Widget.SetFont(calc_btn[i], calc_font); + Gui.Widget.SetCallback(calc_btn[i], "clicked", &@calc_btn_callback); + } + j = 3 + (6 * i); + calc_btn[j] = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, 4, 68 + (36 * i), 32, 32); + calc_btn[j]->color = Color(255, 0, 0); + Gui.Widget.SetText(calc_btn[j], @calc_button_label[j]); + Gui.Widget.SetFont(calc_btn[j], calc_font); + Gui.Widget.SetCallback(calc_btn[j], "clicked", &@calc_btn_callback); + for (k = 1; k < 6; k++) { + calc_btn[j + k] = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, 10 + (36 * k), + 68 + (36 * i), 32, 32); + calc_btn[j + k]->color = T(k == 4 || k == 5 && i == 3, Color(255, 0, 0), Color(0, 0, 255)); + Gui.Widget.SetText(calc_btn[j + k], @calc_button_label[j + k]); + Gui.Widget.SetFont(calc_btn[j + k], calc_font); + Gui.Widget.SetCallback(calc_btn[j + k], "clicked", &@calc_btn_callback); + } + } + Suspend; +} + +Main; \ No newline at end of file diff --git a/Applications/Accessories/Calculator.app/window_icon_16x16.png b/Applications/Accessories/Calculator.app/window_icon_16x16.png new file mode 100644 index 0000000000000000000000000000000000000000..4c07061405ba0d8bbc73b051efef4de5811ca484 GIT binary patch literal 619 zcmV-x0+juUP)B3dD zilQ)D(?-PKo{iPVxyZ?dx+d3V7mWDAy3RmkmFjPY12L6etS`$lH!cp9LgcB4*O zr$!EiY@8}(Cs?NxonS;MR6PJU7QclLhXBR|MFA>Su%cutj+{|)RZrO~P0>JGFP6++ z0CB>dJ|= z$ns{M-&W{sX`r(?2Ee(6yY$XX17MA#4#4`rgUgRrm|xna=#Cy%92S#zhjb2>r=8pR;9*LOL1eU?R$n zSB8>Jw6!(W0qoub1-SWQ1Aw0PR%Tvp0WdU}V)5f90LRj809pqi#zo3jxj5R*`;4Kl zCC2%Y4ps`Do~8(?qbYus9HJnFV(7tN6P>6->1=5rVGU8>NHWe?MD3T+S{N2xTT)R`mWxvw5_GE1lr zHmn8v`>k*9#fS)P(ap#B+00ef)R)!j#xlseYrxj8hu`l*PL^d+I-UM%nED4z(>v9y zeovAlg~baCYsfTTg`-jULLtmfOkg~jMDW^GglhM}@AKpP%nSw-Zz0-6?BBNsqhq7` zi zpzU_6^6l%4hf_z#LIvS4vO_~KGZ|!uh7bsapbaD-+8wY9BK7n>rtY3c#K4iq<$)|` zpsCt3Ns+FiFhOV6l z03HCyFz~xb*8u=s4?&hsl(}G!l?rhWhMZkigf(Y*z`voj1 zEyb?wyYPMbTX}!~o4W!(SCGj-lO$;_H;l^xDu{kTa=@ZBYv6P{v2AN5UsY98%ktcp zem5@y3F&YjYa^%_UP0J?ua|=^b zlXY!vH$Fdk+BZKvJxzvTpm4=X&h2y`&D~|~fWEc5nx9W3K)McQvq7w^#PGm?ah%8o z02f*=REdJ^R367FEy60xvRvzmip9ppMiU)C{7occCc|NRFN%~bDWTWlu-O5`i3rP` zSS-fpYIYykDcJ4Lg2AAe@AYz-*;(uJyo+qo(zq3814^W*=V|KWMg1NKNkDe65zD3%N@77lq=602V4 zJAK&j;DV&0tMk!lgs$=PD76v~0jpQ{x{fiJ&gcr*`h{l{RWpR=gd)@NQrK54l$GXrtedzprR62b zqq4@(i4i7*mlS0-C}0qfT#}h}_66&{MK;Bw$X}&`CXO6=@%lwAOV^C#tmql}3RJAQ z|DUIxot|oRcZHz$)igiH!MP8$s%ie7nr7_`94~NPYWX*1PP5;lS6iCz7~DH>dDGHT z2XM6uBhPA5OV%ZaEPFN!9~Ur`fZ=->SZmCxxzG6%#JA;%?%?nUCZyiI{s&)qXTVsF zGR6P^010qNS#tmY4c7nw4c7reD4Tcy00BWzOjJbx003SyW?VL8S~X`~HDz2hXj(UB zT{dK0H)<XY=HlDr;@jop+~wrl<>cMv<=^M$;pph% z>gnX{>E!L|+9w0>*nn2=k4+B@$v2Q^6v8T@ALEU^z-oc0`BAq0000WbW%=J z06aZDKR`i3-QDBk^z`)f^z`)g_4W1j_4W1j_4W4lkL06m00027Nklb(l7xw{%B3ZTMd@icNkBwc+wf&;Y`EMkN} zB5d-hNg*L2zRGU?CMNtq5w76O@bK^eV@od;B_k~k1_pMqh*%#h9YuEoHB(C-kREnX z7jt!0>o|Q|8-6CRsqDhmW}3S3uFm|-P>a(cJOo(aE@S6&@C+4XMRGekuc0_A gitpID*%)vD03nbW-BHiHaR2}S07*qoM6N<$g13bVYybcN literal 0 HcmV?d00001 diff --git a/Applications/OS/MenuBar.app/Run.HC b/Applications/OS/MenuBar.app/Run.HC new file mode 100644 index 0000000..f828131 --- /dev/null +++ b/Applications/OS/MenuBar.app/Run.HC @@ -0,0 +1,126 @@ +Gui.App(); + +Window* sys_menu = Menu.New("System Menu"); + +U0 @system_menu_item_clicked(MenuItemWidget* widget) +{ + if (!widget->path) + return; + SystemStarter.CreateTask(widget->path, &widget->text); +} + +U0 @system_menu_init() +{ + JsonObject* data = Json.ParseFile("M:/Settings/SystemMenu.json", erythros_mem_task); + JsonArray* items = data->a("items"); + JsonArray* subitems = NULL; + Window* submenu = NULL; + I64 i; + I64 j; + // System Menu supports 2 levels of nesting to separate groups of Applications + // into categories. + for (i = 0; i < items->length; i++) { + if (items->o(i)->@("name")) { + if (items->o(i)->@("path")) { + Menu.AddItem( + sys_menu, items->o(i)->@("name"), + @image_file_to_context2d(items->o(i)->@("icon")), + &@system_menu_item_clicked, + items->o(i)->@("path")); + } + if (items->o(i)->@("items")) { + submenu = Menu.New(items->o(i)->@("name")); + submenu->x = sys_menu->width; + submenu->y = 35 + sys_menu->y + (MENU_ITEM_MIN_HEIGHT * i); + subitems = items->o(i)->a("items"); + for (j = 0; j < subitems->length; j++) { + if (subitems->o(j)->@("name")) { + if (subitems->o(j)->@("path")) { + Menu.AddItem(submenu, + subitems->o(j)->@("name"), + @image_file_to_context2d( + subitems->o(j)->@("icon")), + &@system_menu_item_clicked, + subitems->o(j)->@("path")); + } + } + } + Menu.AddItem( + sys_menu, items->o(i)->@("name"), + @image_file_to_context2d(items->o(i)->@("icon")), + NULL, NULL, submenu); + } + } + } +} + +U0 @system_menu_show(Context2DWidget* widget) +{ + sys_menu->x = 0; + sys_menu->y = 32; + if (widget->width == 30 && widget->height == 30) + Fill2D(widget->ctx, Color(192, 192, 192)); + Gui.Window.Show(sys_menu); + Gui.Window.SetFocus(sys_menu); + Gui.Window.Refresh(sys_menu); +} + +U0 Main() +{ + Window* win = Compositor.CreateWindow(0, 0, Display.Width(), 32, + WIN_FLAGS_NOHILIGHT | WIN_FLAGS_SKIP); + Gui.Window.SetTitle(win, "MenuBar"); + + ButtonWidget* system_menu_btn = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, 0, 0, 58, 24); + Gui.Widget.SetText(system_menu_btn, ""); + system_menu_btn->image = @image_file_to_context2d("system_menu.png"); + system_menu_btn->width = system_menu_btn->image->width + 8; + + Context2DWidget* separator = Gui.CreateWidget( + win, WIDGET_TYPE_CONTEXT2D, + system_menu_btn->x + system_menu_btn->width + 4, 2, 2, 20); + separator->ctx = NewContext2D(2, 20); + Line2D(separator->ctx, 0, 0, 0, 20, Color(224, 224, 224)); + Line2D(separator->ctx, 1, 0, 1, 20, Color(128, 128, 128)); + + TextInputWidget* active_app_title = Gui.CreateWidget( + win, WIDGET_TYPE_LABEL, separator->x + separator->width + 8, 8, 192, 64); + + if (!Compositor.active_win) + Gui.Window.SetFocus(Compositor.GetWindowByTitle("Wallpaper")); + + Gui.Widget.SetFont(active_app_title, "Eight Bit Dragon"); + Gui.Widget.SetText(active_app_title, Compositor.active_win->title); + + CDateStruct ds; + U8 clock_text[32]; + TextInputWidget* clock = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, Display.Width() - 94, 8, 64, 32); + Gui.Widget.SetFont(clock, "Eight Bit Dragon"); + Gui.Window.Refresh(win); + + // FIXME: This is an ugly way to do this... + Compositor.menubar.win = win; + Compositor.menubar.task = Fs; + Compositor.menubar.title = active_app_title; + + Gui.Widget.SetCallback(system_menu_btn, "clicked", &@system_menu_show); + + // System Clock + I64 min = -1; + while (1) { + Date2Struct(&ds, Now); + if (ds.min != min) { + StrPrint(&clock_text, "%02d %03tZ, %02d:%02d", ds.day_of_mon, ds.mon - 1, + "ST_MONTHS", ds.hour, ds.min); + Gui.Widget.SetText(clock, &clock_text); + Gui.Window.Refresh(win); + min = ds.min; + } + if (Compositor.active_win != sys_menu && Gui.Window.IsVisible(sys_menu)) + Gui.Window.Hide(sys_menu); + Sleep(1); + } +} + +@system_menu_init; +Main; \ No newline at end of file diff --git a/Applications/OS/MenuBar.app/system_menu.png b/Applications/OS/MenuBar.app/system_menu.png new file mode 100644 index 0000000000000000000000000000000000000000..345b7ec8331880b46551ed9439861318eb461808 GIT binary patch literal 869 zcmV-r1DgDaP)oBHh($y5sfAF} zGXYxYBFIo$v(3qwnK=KBk6V1I-VCc=|L?0iT9>o72EJzDG0n6b;R)gRnqw;767JLr z>1E*q;jkqK6uwk?#^Wc&d5_-|Tgk)}_*KtP{ugI{u~x8(!h^znMZKi`oJZXxm^J2R zrS{0|#U??FC?PVGSV6~OSgIkJ)%QidebcOpN0Yxv8A}{_3gQimT9U3=sX5WJ^68XS z?tk;NbE6|(_h1wPkFNXL4^7)pt+?*zq3c$Upy2_|jhcT|avFaZy;9Tq_Mv?PPA_U& zW(UqTq5IxUTj_79L8d>KgO?L%OF_pqv@CjA{j76*2g!ALqAS?lg8|v!zVrpuS!eb* zoAYG=000SaNLh0L04^f{04^f|c%?sf0002GP)t-s0002yM>gk1Hs?q<=S)24Sxe|z zOz2xo={YCqIw$E`Oy)r_;T#9xAr0a)AmTG2<}@PfH6r6NALB9~RC(aSxfO_Rq|t1?qO5^|NsC0|Ns8}|NsC0{QLj^|NrLV|Nj2}|NrcCXzhG(?S69e zqMGxlp!BSz^sc7&#lZU7*ZSPq`rX?6=;i;Ei2sy{|FWt7wXFZ*-v9LS|Mm3$`uhL; z`~Up>|Nj2}|Nq*Fm#6>$048)&PE!B?000320RaI40Rtc-dwYC-fPkRB!P?v0^z`}p z`~3X<{r>*`Pz49l0001_Nkl7Fvi{1c_o1K@>!+G!`~NLX2S4X0z@l!X_@e2@-4t8w3O$q0P=;Rlbno&XVY`x z;1dl5XknCYO?U0_L?X_=_V^XZ(aBV0oP~? zFIssa00009a7bBm000fw000fw0YWI7cmMzZ*icMVMV+0Uo}QkdprD_hpPikZoSdA{ z(9qM<(*OVf1qB5U5Dz9NCPPF)UtnEeVPI%zXmoURdwqC_iiL@ZiH?wqkCBXlq(=H}<;=jiC@>FMd}>gwz3>+J08?d|RE z?(XmJ@9^;O@$vEU^78ZZ^Yrxe_V)Jr`uh9(`~3X;{r&y^{{H{}|3{!7wEzGB2y{|T zQyQkGr}_H*{ZfXO82|tPsYygZR2WxXkGqXRK@bGHcV^!cNEkea5Y8h6$ej@IaR>l} z03FB92V*%5+ANhyYC_5q|2u~(Wg{QpyAa)8Yy^NzzyAIlb&@tcvkSAcObV2={WHN~ z%>pP;ELtsy7TE#QSpjhmcEf{}$iJ}KjpkedxVsJAXrUtj4SP2%xKJ1)dxM^GXi#zM z0LJkaC}IgfPa*T`e0T_@EhJ7XbyGg)V%PY6{{>9zPfU2LDH;F(002ovPDHLkV1h!9 ByFLH_ literal 0 HcmV?d00001 diff --git a/Applications/OS/ShutDown.app/Run.HC b/Applications/OS/ShutDown.app/Run.HC new file mode 100644 index 0000000..219f294 --- /dev/null +++ b/Applications/OS/ShutDown.app/Run.HC @@ -0,0 +1,25 @@ +Gui.App(); + +U0 @shutdown_callback(U64 o) +{ + if (!o) + return; + if (o(Window*)->signature == WIN_SIGNATURE) { + Compositor.DestroyWindow(o(Window*)); + Gui.Window.SetFocus(Compositor.GetWindowByTitle("Wallpaper")); + return; + } + switch (o(Widget*)->tag) { + case TRUE: + System.PowerOff(); + break; + default: + Compositor.DestroyWindow(o(Widget*)->parent_win); + Gui.Window.SetFocus(Compositor.GetWindowByTitle("Wallpaper")); + return; + break; + } +} + +MessageBox.Alert("Do you want to shut down this computer?", + "[\"OK\",\"Cancel\"]", &@shutdown_callback); \ No newline at end of file diff --git a/Applications/OS/TaskSwitcher.app/Run.HC b/Applications/OS/TaskSwitcher.app/Run.HC new file mode 100644 index 0000000..10453ed --- /dev/null +++ b/Applications/OS/TaskSwitcher.app/Run.HC @@ -0,0 +1,198 @@ +Gui.App(); +#define MAX_THUMBS_PER_LINE 10 + +Window* win = Compositor.CreateWindow(120, 120, 640, 192, WIN_FLAGS_SKIP); +Window* to_be_focused_win = NULL; +Bool task_switcher_active = FALSE; +Bool task_switcher_invoke = FALSE; + +I64 caption_width = 0; +F64 win_scale = 0.0; +I64 win_thumb_size = 128; +I64 win_thumb_container_size = win_thumb_size + 32; +I64 win_index = -1; +I64 win_items = 0; +I64 cur_index = 0; +U8* cur_win_title = NULL; +Context2D* win_scaled = NULL; + +@compositor_windows_list* win_list = NULL; + +TextLabelWidget* caption = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 8, 8, 256, 64); +Gui.Widget.SetFont(caption, "Eight Bit Dragon"); + +Context2DWidget* cursor = NULL; + +U0 keypress_callback(Window* win, I64) +{ + I64 i; + if (!task_switcher_active) { + if ((KeyDown(SC_CTRL) && KeyDown(Char2ScanCode('w'))) || (KeyDown(SC_ALT) && KeyDown(SC_F4))) + if (Compositor.active_win != win && Compositor.active_win->callback.close) + Compositor.active_win->callback.close(Compositor.active_win); + return; + } + if (KeyDown(SC_TAB)) { + if (KeyDown(SC_SHIFT)) + win_index--; + else + win_index += T(win_items > 1 && task_switcher_invoke, 2, 1); + } + + if (!win_items) + return; + + @window_widgets_list* set_item = win->widget->next->next->next; + if (win_index < 0) + win_index = win_items - 1; + if (win_index >= win_items) + win_index = 0; + for (i = 0; i < win_index; i++) + set_item = set_item->next; + + cursor->x = set_item->widget->x; + cursor->y = set_item->widget->y; + + to_be_focused_win = set_item->widget->id; + Gui.Widget.SetText(caption, set_item->widget->tag); + caption->width = Print2D(NULL, Compositor.theme.font.menu, 0, 0, , , &caption->text); + caption->x = (win->width / 2) - (caption->width / 2); + caption->y = win->height - 24; + + Gui.Window.Refresh(win); + task_switcher_invoke = FALSE; +} + +U0 @taskswitcher_clear_win_thumbs(Window* win) +{ + @window_widgets_list* iter_widget = win->widget; + @window_widgets_list* del_widget = NULL; + while (iter_widget->next) + iter_widget = iter_widget->next; + while (iter_widget->widget->type == WIDGET_TYPE_CONTEXT2D) { + iter_widget = iter_widget->prev; + del_widget = iter_widget->next; + iter_widget->next = NULL; + if (del_widget->widget(Context2DWidget*)->ctx) + DelContext2D(del_widget->widget(Context2DWidget*)->ctx); + Free(del_widget->widget->tag); + Free(del_widget->widget); + Free(del_widget); + } + Gui.Widget.SetText(caption, ""); +} + +U0 @taskswitcher_render_win_thumbs(Window* win) +{ + @compositor_windows_list* origin_list; + @compositor_windows_list* iter_list; + Context2DWidget* widget = NULL; + cursor = Gui.CreateWidget(win, WIDGET_TYPE_CONTEXT2D, -999, -999, + win_thumb_container_size, win_thumb_container_size); + cursor->ctx = NewContext2D(win_thumb_container_size, win_thumb_container_size); + Fill2D(cursor->ctx, Compositor.theme.color.hilight); + + I64 min_caption_width = 0; + win_items = 0; + I64 cur_item = 0; + I64 item_offset = 0; + + origin_list = Compositor.windows->next; + while (origin_list->next) + origin_list = origin_list->next; + + iter_list = origin_list; + while (iter_list) { + if (iter_list->window) + if (!@gui_window_flag_is_set(iter_list->window, WIN_FLAGS_SKIP)) { + min_caption_width = Max(min_caption_width, + Print2D(NULL, Compositor.theme.font.menu, 0, 0, , , + "%s - [%dx%d] at [%d, %d]", iter_list->window->title, + iter_list->window->width, iter_list->window->height, + iter_list->window->x, iter_list->window->y)); + win_items++; + } + iter_list = iter_list->prev; + } + + win->width = Max((win_thumb_container_size * win_items) + T(win_items, 4, 0), + 64 + min_caption_width); + + item_offset = (win->width / 2) - (((win_thumb_container_size * win_items) + T(win_items, 0, 0)) / 2); + + iter_list = origin_list; + while (iter_list) { + if (iter_list->window) + if (!@gui_window_flag_is_set(iter_list->window, WIN_FLAGS_SKIP)) { + widget = Gui.CreateWidget( + win, WIDGET_TYPE_CONTEXT2D, + (win_thumb_container_size * cur_item) + item_offset, 0, + win_thumb_container_size, win_thumb_container_size); + widget->ctx = NewContext2D(win_thumb_container_size, win_thumb_container_size); + Fill2D(widget->ctx, NULL); + win_scale = 1.0; + while (iter_list->window->backing_store->width * win_scale > win_thumb_size || iter_list->window->backing_store->height * win_scale > win_thumb_size) + win_scale -= 0.001; + + win_scaled = Scale2D(iter_list->window->backing_store, win_scale, win_scale); + Fill2D(widget->ctx, NULL); + Blot2D(widget->ctx, + (win_thumb_container_size / 2) - (win_scaled->width / 2), + (win_thumb_container_size / 2) - (win_scaled->height / 2), + win_scaled); + DelContext2D(win_scaled); + widget->tag = CAlloc(256); + StrPrint(widget->tag, "%s - [%dx%d] at [%d, %d]", + iter_list->window->title, iter_list->window->width, + iter_list->window->height, iter_list->window->x, + iter_list->window->y); + widget->id = iter_list->window; // FIXME: We are abusing this property, + // introduce a better way to do this + + cur_item++; + } + iter_list = iter_list->prev; + } +} + +U0 Main() +{ + Gui.Window.SetTitle(win, "TaskSwitcher"); + Gui.Window.EnableAlphaChannel(win); + Gui.Window.SetOpacity(win, 224); + Compositor.RegisterForGlobalInputEvents(win); + Gui.Window.SetCallback(win, "keypress", &keypress_callback); + Gui.Window.Hide(win); + + while (1) { + if (KeyDown(SC_ALT) && KeyDown(SC_TAB)) + task_switcher_active = TRUE; + if (!KeyDown(SC_ALT)) + task_switcher_active = FALSE; + switch (task_switcher_active) { + case TRUE: + if (!to_be_focused_win) { + to_be_focused_win = Compositor.active_win; + task_switcher_invoke = TRUE; + @taskswitcher_render_win_thumbs(win); + Gui.Window.Center(win); + Gui.Window.SetFocus(win); + Gui.Window.Show(win); + } + break; + case FALSE: + if (to_be_focused_win) { + Gui.Window.SetFocus(to_be_focused_win); + Gui.Window.Show(to_be_focused_win); + to_be_focused_win = NULL; + Gui.Window.Hide(win); + @taskswitcher_clear_win_thumbs(win); + win_index = -1; + } + break; + } + Sleep(1); + } +} + +Main; \ No newline at end of file diff --git a/Applications/OS/TempleOS.app/Run.HC b/Applications/OS/TempleOS.app/Run.HC new file mode 100644 index 0000000..ef01123 --- /dev/null +++ b/Applications/OS/TempleOS.app/Run.HC @@ -0,0 +1,127 @@ +Gui.App(); + +U32 tos_palette_std[16] = { + Color(0, 0, 0), Color(0, 0, 170), Color(0, 170, 0), + Color(0, 170, 170), Color(170, 0, 0), Color(170, 0, 170), + Color(170, 85, 0), Color(170, 170, 170), Color(85, 85, 85), + Color(85, 85, 255), Color(85, 255, 85), Color(85, 255, 255), + Color(255, 85, 85), Color(255, 85, 255), Color(255, 255, 85), + Color(255, 255, 255) +}; + +I64 win_mouse_x = ms.pos.x; +I64 win_mouse_y = ms.pos.y; + +CTask* templeos_winmgr_task = NULL; + +U0 @templeos_close_callback(Window* win) +{ + Kill(templeos_winmgr_task); + Compositor.UnregisterForGlobalInputEvents(win); + Compositor.DestroyWindow(win); + win = NULL; + Exit; +} + +Context2D* win_hide_pointer_ctx = NewContext2D(4, 4); + +U0 @templeos_mouseat_callback(Window* win) +{ + win_mouse_x = win->mouse.x - 4; + win_mouse_y = win->mouse.y - 24; + if (win_mouse_x > 0 && win_mouse_x < 640 && win_mouse_y > 0 && win_mouse_y < 480) { + win->pointer = win_hide_pointer_ctx; + } else { + win->pointer = NULL; + } +} + +U0 @templeos_keypress_callback(Window* win, I64 key) +{ + if (win != Compositor.active_win) + return; + if (!KeyDown(SC_GUI) && key) { + if (KeyDown(SC_CURSOR_UP)) { + PostMsg(sys_focus_task, MSG_KEY_DOWN_UP, 0, SC_CURSOR_UP); + return; + } + if (KeyDown(SC_CURSOR_DOWN)) { + PostMsg(sys_focus_task, MSG_KEY_DOWN_UP, 0, SC_CURSOR_DOWN); + return; + } + if (KeyDown(SC_CURSOR_LEFT)) { + PostMsg(sys_focus_task, MSG_KEY_DOWN_UP, 0, SC_CURSOR_LEFT); + return; + } + if (KeyDown(SC_CURSOR_RIGHT)) { + PostMsg(sys_focus_task, MSG_KEY_DOWN_UP, 0, SC_CURSOR_RIGHT); + return; + } + // FIXME: Ctrl-key combinations + if (KeyDown(SC_SHIFT)) { + XTalkWait(sys_focus_task, "%c", SHIFT_KEY_SCAN_DECODE_TABLE(U8*)[key]); + } else { + XTalkWait(sys_focus_task, "%c", NORMAL_KEY_SCAN_DECODE_TABLE(U8*)[key]); + } + } +} + +Context2D* icon = Image.FileToContext2D("window_icon_16x16.png"); + +Window* win = Compositor.CreateWindow( + 18, 554, 649, 508, + (WIN_FLAGS_MOVABLE | WIN_FLAGS_ICON | WIN_FLAGS_TITLE_BAR | WIN_FLAGS_MIN_BUTTON | WIN_FLAGS_CLOSE_BUTTON), + "TempleOS Window Manager", icon); + +U0 Main() +{ + Gui.Window.Refresh(win); + Compositor.RegisterForGlobalInputEvents(win); + Gui.Window.SetCallback(win, "close", &@templeos_close_callback); + Gui.Window.SetCallback(win, "keypress", &@templeos_keypress_callback); + Gui.Window.SetCallback(win, "mouseat", &@templeos_mouseat_callback); + + Gui.Window.SetFocus(win); + + while (win) { + if (win == Compositor.active_win) { + win->flags |= WIN_FLAGS_NOFILL; + if (IsSuspended(sys_winmgr_task)) { + keydev.fp_ctrl_alt_cbs = tos_fp_cbs_enabled; + Suspend(sys_winmgr_task, FALSE); + } + ms.pos.x = win_mouse_x; + ms.pos.y = win_mouse_y; + } else { + if (!IsSuspended(sys_winmgr_task)) { + keydev.fp_ctrl_alt_cbs = tos_fp_cbs_disabled; + Suspend(sys_winmgr_task, TRUE); + } + } + Sleep(1); + } +} + +U0 @templeos_winmgr_redraw() +{ + CDC* dc; + I64 x, y; + while (1) { + dc = DCScrnCapture; + for (y = 0; y < GR_HEIGHT; y++) { + for (x = 0; x < GR_WIDTH; x++) { + Plot2D(win->render_ctx, x + 4, y + 24, + tos_palette_std[dc->body[(y * dc->width) + x]]); + Plot2D(win->backing_store, x + 4, y + 24, + tos_palette_std[dc->body[(y * dc->width) + x]]); + } + } + DCDel(dc); + Sleep(1); + } +} + +templeos_winmgr_task = Spawn(&@templeos_winmgr_redraw, , , 3); +// Adam("blkdev.boot_drv_let='C';WinFocus(User(\"WinMax;\n\"));\n"); + +Main; \ No newline at end of file diff --git a/Applications/OS/TempleOS.app/window_icon_16x16.png b/Applications/OS/TempleOS.app/window_icon_16x16.png new file mode 100644 index 0000000000000000000000000000000000000000..d6f6461538200858463e73c227b3521254dc40f1 GIT binary patch literal 268 zcmeAS@N?(olHy`uVBq!ia0y~yU=RRd7G?$phPQVgfdn`UJR*x37`TN&n2}-D90>*n z2KEw9Usv`gtUTg8>X#q(3(h4oFfcHd1o;IsI6S+N2I3@n zySuQ9C3d`KU|?|Zba4#fkWEeqVq#`vQxg{`XfkqHz;j%L!#O}eNmWCGW1>-^pkbqj zrBjJzmxYy6Y3G46jfOIbYV3z3lnmI;1g$V;c;6?Q`8OkCCddVzu6{1-oD!MMU`)%&240000NbVXQnL3MO!Z*l-k zVQ_CCLt$)TZDDR?a%E}Xks2xh001d;MObu1WpiV4X>fFDZ*BlWVRL08 zf5DUiA*F()AYltZSQ0F(MXe9A~(p8hgrCcID3cU7{!=j$^_wB#NSgR!X6@{-evX z-9*pl&ZGmdCt<(Ws>G8n&}KPX-3v+0pOfNq@Gp~#xc)dzF>WA zjZz!t7dmKDF&q_)Mng(nP!z=kKv0u_Gqba--dtt#$z$d_3(TFJqpB<_2ngbs$XXIB zi2DaXSB85J?o*VScDv2@!2oM3N^K~M0%J5a4)FdC2*MB*D4J=?^{dx7on<&+``Hdj zl3;CxF@~>S`jdb|6UEgi&%^m1uim`I_k93-rKqtW9^>4OO`KBDYPE=xm@te8LPevY zexJ-z8H+I*U6k~v6&wQgKkOelNK*i*g{K=EM~@*O^{*g#%U70n#oA*&liKr6`p!QA X-7_S$@ueDA00000NkvXXu0mjf^;IB? literal 0 HcmV?d00001 diff --git a/Applications/OS/Terminal.app/Run.HC b/Applications/OS/Terminal.app/Run.HC new file mode 100644 index 0000000..177b111 --- /dev/null +++ b/Applications/OS/Terminal.app/Run.HC @@ -0,0 +1,205 @@ +Gui.App(); + +Window* win = Compositor.CreateWindow(18, 62, 684, 474, WIN_FLAGS_DEFAULT); +Gui.Window.EnableAlphaChannel(win); +Gui.Window.SetOpacity(win, 224); +Gui.Window.SetIcon(win, Image.FileToContext2D("Icon.png")); +Gui.Window.SetTitle(win, "Terminal"); + +TerminalWidget* active_term = NULL; +@shell* sh = NULL; + +U0 @terminal_keypress_callback(Window* win, I64 key) +{ + if (!active_term || Compositor.active_win != win) + return; + I64 i; + U8 send_key[4]; + MemSetU32(&send_key, 0, 1); + switch (key) { + case SC_CURSOR_UP: + send_key[0] = '\x1b'; + send_key[1] = '['; + send_key[2] = 'A'; + break; + case SC_CURSOR_DOWN: + send_key[0] = '\x1b'; + send_key[1] = '['; + send_key[2] = 'B'; + break; + case SC_CURSOR_LEFT: + send_key[0] = '\x1b'; + send_key[1] = '['; + send_key[2] = 'D'; + break; + case SC_CURSOR_RIGHT: + send_key[0] = '\x1b'; + send_key[1] = '['; + send_key[2] = 'C'; + break; + case SC_DELETE: + send_key[0] = 21; + break; + case SC_BACKSPACE: + send_key[0] = 8; + break; + case SC_TAB: + send_key[0] = 9; + break; + case SC_ENTER: + send_key[0] = 10; + break; + case SC_HOME: + send_key[0] = 22; + break; + case SC_END: + send_key[0] = 23; + break; + case SC_PAGE_UP: + send_key[0] = 24; + break; + case SC_PAGE_DOWN: + send_key[0] = 25; + break; + case SC_ESC: + send_key[0] = 27; + break; + case 0x02 ... 0x0D: + case 0x10 ... 0x1B: + case 0x1E ... 0x29: + case 0x2B ... 0x35: + case 0x39: + if (!KeyDown(SC_SHIFT)) { + if (KeyDown(SC_CTRL)) { + switch (key) { + case Char2ScanCode('c'): + send_key[0] = 0x03; + sh->break = TRUE; + break; + default: + break; + } + } else { + send_key[0] = NORMAL_KEY_SCAN_DECODE_TABLE(U8*)[key]; + } + } else { + if (key == 0x39) // Handle TempleOS SHIFT-SPACE character. + send_key[0] = ' '; + else { + if (KeyDown(SC_CTRL)) { + // terminal copy paste handling + } else { + send_key[0] = SHIFT_KEY_SCAN_DECODE_TABLE(U8*)[key]; + } + } + }; + break; + default: + return; + break; + } + for (i = 0; i < 4; i++) + if (send_key[i] && active_term->output) + FifoU8Ins(active_term->output, send_key[i]); +} + +TerminalWidget* t = Gui.CreateWidget(win, WIDGET_TYPE_TERMINAL, 0, 0, 120, 120); +t->pointer = Compositor.theme.pointer.text; +VerticalScrollBarWidget* vscroll = Gui.CreateWidget(win, WIDGET_TYPE_VERT_SCROLLBAR, 0, 0, 16, 128); +vscroll->height = 128; +Context2DWidget* status = Gui.CreateWidget(win, WIDGET_TYPE_CONTEXT2D, 0, 0, Display.Width(), 44); +status->ctx = NewContext2D(Display.Width(), 44); +Fill2D(status->ctx, Color(204, 204, 204, win->opacity)); + +U0 @terminal_create_new_instance() +{ + U32 init_bg_color = t->color.background; + init_bg_color.u8[3] = win->opacity; + Fill2D(t->backing_store, init_bg_color); + active_term = t; + sh = @shell_new; + sh->session = &Compositor.session; + t->output = sh->input; + sh->output = t->input; +} + +U0 window_close(Window* win) +{ + if (sh) + if (sh->task && sh->exit) + Free(sh); + if (win == Compositor.active_win) + Gui.Window.SetFocus(Compositor.GetWindowByTitle("Wallpaper")); + Compositor.UnregisterForGlobalInputEvents(win); + Compositor.DestroyWindow(win); +} + +U0 @terminal_vscroll_change(Widget*) +{ + I64 i = 0; + I64 max_scroll = vscroll->height - 32; + F64 f1 = (ToF64(max_scroll) / ToF64(t->cursor.y + t->max.y - 2)); + while (vscroll->scroll > ToI64(i * f1)) + i++; + t->scroll.y = i; + t->refresh = TRUE; + Gui.Window.Refresh(win); +} + +U0 Main() +{ + Compositor.RegisterForGlobalInputEvents(win); + Gui.Window.SetCallback(win, "keypress", &@terminal_keypress_callback); + Gui.Widget.SetCallback(vscroll, "change", &@terminal_vscroll_change); + I64 prev_width = -1; + I64 prev_height = -1; + I64 prev_max_x = -1; + I64 prev_max_y = -1; + F64 f1; + Gui.Window.SetFocus(win); + Gui.Window.SetCallback(win, "close", &window_close); + + @terminal_create_new_instance; + + while (win) { + // FIXME: This should be event-driven... + if (win->width != prev_width || win->height != prev_height) { + win->width = 3 + RoundI64(win->width, 8); + win->height = RoundI64(win->height, 16); + prev_width = win->width; + prev_height = win->height; + if (active_term) { + active_term->width = win->width; + active_term->height = win->height; + } + status->y = win->height - 44; + goto terminal_update_vscroll; + } + if (prev_max_x != t->max.x || prev_max_y != t->max.y) { + terminal_update_vscroll: + if (!t->max.y) { + vscroll->x = Display.Width(); // Hide + } else { + vscroll->x = win->width - vscroll->width - 9; + vscroll->height = win->height - 44; + f1 = (ToF64(t->size.rows) / ToF64(1 + t->cursor.y + t->max.y)); + vscroll->length = ToI64((vscroll->height - 32) * f1); + vscroll->scroll = vscroll->height; + } + prev_max_x = t->max.x; + prev_max_y = t->max.y; + // System.Log(Fs, "vscroll->scroll: %d", vscroll->scroll); + + Gui.Window.Refresh(win); + } + if (FifoU8Cnt(active_term->input)) + Gui.Window.Refresh(win); + if (sh->exit) { + win->callback.close(win); + return; + } + Sleep(10); + } +} + +Main; \ No newline at end of file diff --git a/Applications/OS/Wallpaper.app/Run.HC b/Applications/OS/Wallpaper.app/Run.HC new file mode 100644 index 0000000..d07d5a2 --- /dev/null +++ b/Applications/OS/Wallpaper.app/Run.HC @@ -0,0 +1,29 @@ +// Gui.App(); + +U0 @event_loop(CTask* task) +{ + Fs->ipc = task->ipc; + IpcMessage* msg; + while (1) { + msg = Ipc.MsgRecv(); + if (msg) { + Free(msg); + } + Sleep(1); + } +} + +Ipc.InitQueue(Fs); +Spawn(&@event_loop, Fs); + +U0 Main() +{ + System.Log(Fs, "Task running at 0x%08x", Fs); + Window* win = Compositor.CreateWindow(0, 0, Display.Width(), Display.Height(), + WIN_FLAGS_NO_REINDEX | WIN_FLAGS_SKIP, + "Wallpaper"); + Compositor.SetWallpaper(Compositor.theme.wallpaper); + Suspend; +} + +Main; \ No newline at end of file diff --git a/Applications/TestApplication.app/Run.HC b/Applications/TestApplication.app/Run.HC new file mode 100644 index 0000000..8539dfe --- /dev/null +++ b/Applications/TestApplication.app/Run.HC @@ -0,0 +1,96 @@ +#include "M:/Include/Gui"; + +U0 window_close(Window* win) { Compositor.DestroyWindow(win); } + +U0 btn1_click(Widget* widget) +{ + no_warn widget; + Window* win = Compositor.CreateWindow(Rand * Display.Width(), Rand * Display.Height(), + 320, 240, WIN_FLAGS_DEFAULT); + win->callback.close = &window_close; + Gui.Window.SetTitle(win, "New Window"); + Gui.Window.SetFocus(win); +} + +U0 btn2_click(Widget*) +{ + System.text_mode = TRUE; + Dbg; +} + +U0 hs1_change(HorizontalSliderWidget* widget) +{ + Gui.Window.SetOpacity(widget->parent_win, + ClampI64(1.4 * widget->scroll, 0, 255)); +} + +U0 Main() +{ + Window* win = Compositor.CreateWindow(240, 240, 662, 504, WIN_FLAGS_DEFAULT); + win->alpha = TRUE; + Gui.Window.Center(win); + Gui.Window.SetTitle(win, "Test Application"); + + TextInputWidget* label1 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 8, 0, 192, 64); + Gui.Widget.SetText(label1, + "\n" + "Welcome to Erythros desktop environment for TempleOS!\n" + "\n" + "This is a test application for Gui Widgets. " + "There will be bugs. Please report them! :)\n" + "\n"); + + ButtonWidget* btn1 = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, 8, 56, 192, 32); + Gui.Widget.SetText(btn1, "Click me for a new Window"); + + ButtonWidget* btn2 = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, 224, 56, 256, 32); + Gui.Widget.SetText(btn2, "Click me to open TempleOS Debugger"); + + btn1->callback.clicked = &btn1_click; + btn2->callback.clicked = &btn2_click; + + CheckBoxWidget* cb1 = Gui.CreateWidget(win, WIDGET_TYPE_CHECKBOX, 8, 160, 14, 14); + TextInputWidget* label2 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 32, 162, 100, 14); + Gui.Widget.SetText(label2, "Some random checkbox"); + + RadioButtonWidget* radio1 = Gui.CreateWidget(win, WIDGET_TYPE_RADIO, 192, 160, 14, 14); + RadioButtonWidget* radio2 = Gui.CreateWidget(win, WIDGET_TYPE_RADIO, 192, 180, 14, 14); + RadioButtonWidget* radio3 = Gui.CreateWidget(win, WIDGET_TYPE_RADIO, 192, 200, 14, 14); + TextInputWidget* label3 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 216, 162, 192, 14); + Gui.Widget.SetText(label3, "Radio #1"); + TextInputWidget* label4 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 216, 182, 192, 14); + Gui.Widget.SetText(label4, "Radio #2"); + TextInputWidget* label5 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 216, 202, 192, 14); + Gui.Widget.SetText(label5, "Radio #3"); + + Gui.Widget.SetEcho(label2, cb1); + Gui.Widget.SetEcho(label3, radio1); + Gui.Widget.SetEcho(label4, radio2); + Gui.Widget.SetEcho(label5, radio3); + + HorizontalSliderWidget* hs1 = Gui.CreateWidget(win, WIDGET_TYPE_HORZ_SLIDER, 338, 192, 192, 64); + hs1->max = 100; + hs1->scroll = (hs1->width / 5) * 4.25; + hs1->callback.change = &hs1_change; + TextInputWidget* label6 = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 398, 220, 192, 64); + Gui.Widget.SetText(label6, "Window Opacity"); + + VerticalSliderWidget* vs1 = Gui.CreateWidget(win, WIDGET_TYPE_VERT_SLIDER, 576, 192, 64, 192); + vs1->max = 100; + vs1->scroll = hs1->height / 2; + + Gui.Widget.SetEcho(vs1, hs1); + + TextInputWidget* input1 = Gui.CreateWidget(win, WIDGET_TYPE_INPUT, 8, 320, 256, 64); + input1->font = BitmapFonts.GetByName("Eight Bit Dragon"); + Gui.Widget.SetText(input1, "this is some sample text"); + + Gui.Window.SetFocus(win); + Gui.Window.Refresh(win); + + Gui.Window.SetOpacity(win, ClampI64(1.4 * hs1->scroll, 0, 255)); + + Suspend; +} + +Main; \ No newline at end of file diff --git a/Include/Gui.HC b/Include/Gui.HC new file mode 100644 index 0000000..4b13fd3 --- /dev/null +++ b/Include/Gui.HC @@ -0,0 +1,53 @@ +U0 @gui_event_loop(CTask* task) +{ + Fs->ipc = task->ipc; + IpcMessage* msg; + while (1) { + msg = Ipc.MsgRecv(); + if (msg) { + switch (msg->type) { + // TODO: + case CPZ_MSG_WIN_WIDGET_DESTROY: + @gui_widget_destroy(msg->payload); + case CPZ_MSG_WIN_MOVE_TO: + break; + case CPZ_MSG_WIN_KEY_PRESS: + if (msg->payload(Window*)->callback.keypress) + msg->payload(Window*)->callback.keypress(msg->payload, msg->i64); + @gui_window_repaint(msg->payload, msg->type); + Compositor.theme.window_repaint(msg->payload, msg->type); + if (msg->payload(Window*)->callback.repaint) + msg->payload(Window*)->callback.repaint(msg->payload); + break; + case CPZ_MSG_WIN_MOUSE_AT: + if (msg->payload(Window*)->callback.mouseat) + msg->payload(Window*)->callback.mouseat(msg->payload); + @gui_window_repaint(msg->payload, msg->type); + Compositor.theme.window_repaint(msg->payload, msg->type); + if (msg->payload(Window*)->callback.repaint) + msg->payload(Window*)->callback.repaint(msg->payload); + break; + case CPZ_MSG_WIN_MOUSE_WHEEL: + case CPZ_MSG_WIN_LEFT_BTN_UP: + case CPZ_MSG_WIN_LEFT_BTN_DOWN: + case CPZ_MSG_WIN_RIGHT_BTN_UP: + case CPZ_MSG_WIN_RIGHT_BTN_DOWN: + case CPZ_MSG_WIN_REPAINT: + @gui_window_repaint(msg->payload, msg->type); + Compositor.theme.window_repaint(msg->payload, msg->type); + if (msg->payload(Window*)->callback.repaint) + msg->payload(Window*)->callback.repaint(msg->payload); + break; + // FIXME: add CPZ_MSG_WIN_RESIZE + default: + break; + } + Free(msg); + } + Sleep(1); + } +} + +Ipc.InitQueue(Fs); +Spawn(&@gui_event_loop, Fs, Fs->task_name, 1); +System.Log(Fs, "Task running at 0x%08x", Fs); \ No newline at end of file diff --git a/Media/Sounds/Beep.wav b/Media/Sounds/Beep.wav new file mode 100644 index 0000000000000000000000000000000000000000..6b421b7be4ed904c7ece735c3cb90ada0da90e61 GIT binary patch literal 15404 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 IM6N<$f`wpA8~^|S literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/BitmapFont/monospace.png b/Media/Themes/Umami/BitmapFont/monospace.png new file mode 100644 index 0000000000000000000000000000000000000000..dd0e181a099b19c8d36af62b43df45046fd3ca6f GIT binary patch literal 1371 zcmZ{jc}&v>9L2wF5jn&Hg5Vsajzi=umr8-j(F&ylX<>i`Cs0}`f~EAwqzYJ2kV8=> z78JuNqK-pF5voxfa*4_Y#dEcaAO$xqBDP|o3rn_Sf9xgikN1*K-e0dUG$cSz=PMlm z&?5&DX#i*?0O+GJ0APOibvpnmmF@2z%4RSCY)aEhy(!(HM)}fnK_ojfM4#Iq+EbfX zXH!By>_`e~jvck?45qImcLx_4Mc(PDeTfq(-IZ9xIL27R*MJzK!el$p#yW#*NzZtl z9$3!9{T+%JpP;q(k$r{&cBfZm(7m5OyL)^r0ZTfsf?C0FN^oH#N2>!ap35jG{z3AV z(Cg#NGu4+_R53U1Eu1VIiEeSoIF?ayvJ5?=%{?-LD&*v~%&V+BWV7Y5RfCS`pJW?M zo=2rHBF3y|EV*Jg+a|s__rxVuvwfI(gSY1L?Prf4T48LQ{EMJ?GqafW=xKSPQW$ED zt#3VQyV;}2rgYQRq&!hlS^BU$UGrl6JN4w0Y$PmOk!mQ*n!7o~#a+dUcULL)PxI$o zaY}Sdw8Mc@Z`>5Y`nl+afl1+G6bi6hBc{`}yh)0KhQMYe#fsM7W;H;6O3hb$S`H_VKH*R??bEP=qWP3xDt8t_tq};+=vXq1TGQCjFu|`r&a|;a@=WXl(ZA}F#??n{XblQQQj)$2;}Ya^c3AAvJHDS z&!RvpwV*c3C>KYk7x|6Gt&6JqV+w~$YH54rc`IKXt$Ta7;F)~DHQQ1vC|6)6JvVuN z=X?2$X^d*3V7=<;>>5?@`T+^84WbX}Q@v&pGzFufM9_v_4fmidcn zWgFJ(CYJEE@PDOIkN$1SBAidYUUSWL7f4flm$*j%_b!O=im_HfnWn^z`dD4G5R z-yr0sK6Y~ZNl zCXSWHWG3?)Ps?)w!0aC)fyrZ}u~QK?XAkgb0e-mnbapc0z~1vZZ=h%CXJk?cvBfVo F_b)GjQz!rc literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/BitmapFont/sans.png b/Media/Themes/Umami/BitmapFont/sans.png new file mode 100644 index 0000000000000000000000000000000000000000..244ab5fbc967b9ba700d4c033ea18fa70fa35730 GIT binary patch literal 606 zcmV-k0-^nhP)Px#2~bQ_MF0Q*00030002Gv-7NqB00DGTPE!Ct=GbNc00H(%L_t(IjlGkx${aZm zgsYYkY*-DWe9wWOqXj*=1~kEC7o6Y2_{km2?45u3ec?3?l3GF^rE1~%df;RJ`PcKy z^ZV=ZTCa!h;@__a{`mIw^?cy_^Vj^i`d$9O7rX&??HOyO9Q~k4d-YsBS3l^WgMGpP3?@va@OikDIa5L@O8{~}AW#^ta>x=A z*9qG$2OTi!K+xPTq=V!=fp5GF10V4DD?f$dQ=pW!p{*Q9kg9E81&4eV5+apI!R3Gh zCL9QvvE6uFcx541 zIdrQBQsnto&s+V4iJl~>5lV}6z;TLnXlW=*S<05_pa)3;7|4BPNqT#q1`Nn_4M5Ji zy$5XE_Fi}geHEz7dyU<;*|wk$h|QP!EX~>y9-if)(F9;$9qydwyYtf-1_p*Z1s2G* zAXGw0^6GqJIoe=9XfNZi7qLj%?(_J^dxx@ya?G8TEw^U3%|8148NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;JpsQj7}P}D;O9UssnsNTp5snN$L`l)FpXyF6Ga?R5%*nDf!mRnP{-JZJr_WXShmK=Pv^x&gaM;>oF^>p*;r<>0_-E!vX*3-|oo_V_U z%(Ja$o^LzzY}?sq+s{4Qec}1;i_iC5e7^7U%LA8RTz>HW|NsAHD>t2FU|?V^3GxeO zU}R!Gb6`!zBL)Tr#w2fd7md9ewr*lzV94}zaSYKopPV2e)6hC~D(h;N)jcBGS5;IZ zH{Rf82@*=+y2-jG>y(z(DxpVSU0j@#wU}&go jV5Vzm9%5)@Wn^w;Xav#lJThhpD9Als{an^LB{Ts51AWH* literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/address-book-new.png b/Media/Themes/Umami/Icon/actions/address-book-new.png new file mode 100644 index 0000000000000000000000000000000000000000..1cdd0b6b73d4d00e6e426964218b6554a4b2f2b5 GIT binary patch literal 1083 zcmV-B1jPG^P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0Q>;y08cd6u?+wK010qNS#tmY4c7nw4c7reD4Tcy00TWqL_t(Y$F-HsZyQAz z#(%RP_1F3{P7R5PNSjt9ky@lmg#-TqpdJuX4i(}A+~H3^NL=^>;M7YMju91t1K*X1 zK&lYJjpZhkki>D~xQ^@fuH)U=<=_wO#I9Wl9%(eAcV?bv=6&9sh5s1>z&*I314}>V zE_`tB&u4CKmZqp%0sQv##E{Zm{ruy%-vV5~2hcimmo^XP?_68Ho_*!zi#)0p*gSfH z<7R@I>9kKGx+fC!>vo5YW{W~@ZKd)DE3V|z%vIj)7g$Ip7qiCHt000T6AI7EJilx& zb8Jp_`b;VUKz#ZJwes3KFd(4U9arLeHbPNwPWExVU?)XB0E**c6w3r_4TSV?!a?e@ zX&fW`L}8`fIqP4pTiCU4IC=aPM@Ji2GK=d4+G%hl$<1AyNQm;%Jcm~nu+wppTYKoH zJzhY|tl2;1&a-DOvtNr+C@o^k=X&}^se+Y`QsbK02RbQ7`5f#=_O5jLAVlMRZ&kI0<{Lh z_X#{UX+>Gq)6wt@fCsaJRe=bZJm_){;m3S-iqwN0>a)`v zUb%>`s+copi7$K|<@G#V-LUv%n80v)aKk~W3t5c4GO?Y*b}XP%S(;}Yv|+(e@Z6c0 z6A4khJU5nbtOo*!MnX?bG}#S&&~=SO!l3E6WRekLp#Y;ERdIT}m1ApT0}DPJDquEj zzPNK2P4V&kc5_`Hp$LSa(*!~Yluj=IDLww$E&Hw4-V?Wfzk40HKUCnmZ-3>b#R!Tr z`F}uB6mI=chETk_$)m4Y1_W$ZV#Ly9XEGxRg*Yqz;)6QB6?xnDOXp% zUpKu=^OmA(eg2Q}U~VQssd(QnkrK2EM-1Dlz4`uUt9l^x#>Dv%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vfD=;uRFfi7!MmYcg002ovPDHLkV1j)9 B)t3MO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/appointment-new.png b/Media/Themes/Umami/Icon/actions/appointment-new.png new file mode 100644 index 0000000000000000000000000000000000000000..1c55f86f3fb5bdd09fc211bf84a9623574f81cfb GIT binary patch literal 1477 zcmV;$1v>hPP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00hTLL_t(Y$Hi4^Op{j_ z7GgHd+|*r?Z7xHz#o=NWcVSC*iCZ$Ai)4WUD-IW&Y*{uWOlZ-x;8+%&OZ@K^ll@VY zK?Fs)h*%&Xf?K&(N@*#*(F-lTmx5xSz3-8dFxbqpAAZS`@5?#g`#kUaocElwV#R}g zWNC`Lq70=*maYuR)08p7-Qsj*uqa*8MRWWA7crAoo~VMl{!ff_Xfb+yM1a3}jOU8e z6rSVX6i5-;xE+yApJ4j=cYv2y{VREqM0x(?;ZxA42BE6EfpcdrqpIpMnwnI&bg3C< zD{ArEuT^NMyoSz8?XlX^U%Gs@Y*# zJsW}3Pyw6n9PIik%*;gL@wm~}+KGk>hhfm{L|`%_HXHiRkLV5u3Dr?G+m~jZt|;Kl znY^~90ijTU0{_6cx*Vqd7V_v zNRl3*q`V~p@{H0b+fq~Ah)CE68iLb!9_IdIBvOjet^*iWNf8YCiF}Hxa3T_{z@1Po zqQRdq8@Ym@OF2U}4UtW|7m*icDNeFAs=DhK8#5vluw!~UEJQ2=$6)9_0A2fDXxly) zmHGW%Oij7r^VA?3`VqH7dGI^3VY#&{*3{l%+-AQA~7;G2MM zxLb&fwC6x`HI)Q%770vF+TnGdMR2MdR>M(rT|SMHJ#zgLdk$qCxx?+(($Xd@V6aGZ z+7F{%Ekp+D-^Eb#?gaslXA&-#19rO&Hrpurd)3giY9Ny+XBLs~lNPU(rz^$c>PPjs zH9R1)Vu#IGuK~bd+KSF|4__Ph_+q|1PjDE?}{k(cDrG zi`68$g+;>QAX(uQCxYA7|lY)T3ix?;8ZysY?ICCa9 zo5x`^8feamwu>F&e9D)UR4x(tVvqbKd1l#N-e?28LxPM+ILpSydhurP7+1u$#L4}B zABOsd#n~mxD4iqA*DiYwz{VKKa zG*Kli$Dgoz)gv1UKUI3;huye=J1H>lwFPY**P!mZNupL^RUqJp?v`Hs%Q4@k$sEM9 zmGH!J0uK{5Ccn5XvrwuG$B%1WuMU=R3+;<3czs@xvCrp&c2Ea(YoGXs|IL@k{p;60 znN1vT5T0F5Ad#?v@H$nJwl6hbPiqpp--Sb-$Fe`%Kb)ADm`kxc39k^={L@|Y5LZEX zlaRVKIpyg7v_iW)=ZCwp%rf4g_mEMFB5B!NZfd?`{o1FL^luK~Bf_hM$L@Dq@&c_V zY$3c&*!Ad|b$iLW@85gt<1;DS-v1@(xy{OigoM2`Cn4-0q!6AVto*N6Ggrhd_w;;a zTjue?yq%BFuib-t{sJ4mQKv9DxJm#303~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_Y zR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih) fIx;spF)=GJFgh?W*0Dx800000NkvXXu0mjfM8Bv> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/appointment.png b/Media/Themes/Umami/Icon/actions/appointment.png new file mode 100644 index 0000000000000000000000000000000000000000..1c55f86f3fb5bdd09fc211bf84a9623574f81cfb GIT binary patch literal 1477 zcmV;$1v>hPP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00hTLL_t(Y$Hi4^Op{j_ z7GgHd+|*r?Z7xHz#o=NWcVSC*iCZ$Ai)4WUD-IW&Y*{uWOlZ-x;8+%&OZ@K^ll@VY zK?Fs)h*%&Xf?K&(N@*#*(F-lTmx5xSz3-8dFxbqpAAZS`@5?#g`#kUaocElwV#R}g zWNC`Lq70=*maYuR)08p7-Qsj*uqa*8MRWWA7crAoo~VMl{!ff_Xfb+yM1a3}jOU8e z6rSVX6i5-;xE+yApJ4j=cYv2y{VREqM0x(?;ZxA42BE6EfpcdrqpIpMnwnI&bg3C< zD{ArEuT^NMyoSz8?XlX^U%Gs@Y*# zJsW}3Pyw6n9PIik%*;gL@wm~}+KGk>hhfm{L|`%_HXHiRkLV5u3Dr?G+m~jZt|;Kl znY^~90ijTU0{_6cx*Vqd7V_v zNRl3*q`V~p@{H0b+fq~Ah)CE68iLb!9_IdIBvOjet^*iWNf8YCiF}Hxa3T_{z@1Po zqQRdq8@Ym@OF2U}4UtW|7m*icDNeFAs=DhK8#5vluw!~UEJQ2=$6)9_0A2fDXxly) zmHGW%Oij7r^VA?3`VqH7dGI^3VY#&{*3{l%+-AQA~7;G2MM zxLb&fwC6x`HI)Q%770vF+TnGdMR2MdR>M(rT|SMHJ#zgLdk$qCxx?+(($Xd@V6aGZ z+7F{%Ekp+D-^Eb#?gaslXA&-#19rO&Hrpurd)3giY9Ny+XBLs~lNPU(rz^$c>PPjs zH9R1)Vu#IGuK~bd+KSF|4__Ph_+q|1PjDE?}{k(cDrG zi`68$g+;>QAX(uQCxYA7|lY)T3ix?;8ZysY?ICCa9 zo5x`^8feamwu>F&e9D)UR4x(tVvqbKd1l#N-e?28LxPM+ILpSydhurP7+1u$#L4}B zABOsd#n~mxD4iqA*DiYwz{VKKa zG*Kli$Dgoz)gv1UKUI3;huye=J1H>lwFPY**P!mZNupL^RUqJp?v`Hs%Q4@k$sEM9 zmGH!J0uK{5Ccn5XvrwuG$B%1WuMU=R3+;<3czs@xvCrp&c2Ea(YoGXs|IL@k{p;60 znN1vT5T0F5Ad#?v@H$nJwl6hbPiqpp--Sb-$Fe`%Kb)ADm`kxc39k^={L@|Y5LZEX zlaRVKIpyg7v_iW)=ZCwp%rf4g_mEMFB5B!NZfd?`{o1FL^luK~Bf_hM$L@Dq@&c_V zY$3c&*!Ad|b$iLW@85gt<1;DS-v1@(xy{OigoM2`Cn4-0q!6AVto*N6Ggrhd_w;;a zTjue?yq%BFuib-t{sJ4mQKv9DxJm#303~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_Y zR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih) fIx;spF)=GJFgh?W*0Dx800000NkvXXu0mjfM8Bv> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/back.png b/Media/Themes/Umami/Icon/actions/back.png new file mode 100644 index 0000000000000000000000000000000000000000..90ecc8b397791e29e3ed1c811a076983119527e5 GIT binary patch literal 1135 zcmV-#1d#iQP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00T@(L_t(Y$L*C(XkA4Z zhM)O4=QKB|V%iFct(byFM5uzs~OWzN9N`+nd1&6$D! zqe+OhZs35+7M!|LtZwrqR- zg~3A^CAD2`QcbgXaG2I&QIGfBu_JeX&*SSn?XStwYYOzgL}$tBZwBt&({*=upEoo2 z6}`6%A~n@;Ntd$+8`lfTqStCV)4lmuf$WHPmlW#c;Ld$FbZpwJCTG6@9FdK9dH*{t zQb+^>A(mZ!W$gKOFdCW&^bM)q89(#UqdT5$lYwJ$_H(3ADCzO_nWIRdkwT)RLLd=B zklXs?>MY6*>!Iu0+F#u>us?I5Jj<_Vr%}>FOAqTT)n0 z^pB{+9UC@3w=4HV=FEj3nY;YUGO>&BgvF{VRbhx+4IF5E0<|9FYB=YbR3Pa@A{-Gl zQz0}ZYC_>~C?CKeaY!W)vYxB64p)Jay>wNJ8vcVHTB4`s&!0Q;>BI-cTQ~I3edB$E zri7IhP*rpQev2Z=DAu(qg4X)C*CMf^+FXE9?5Q`*(8a||Bgao1Ew%ew=8^`M}HQ6XK!RC4i~Nzy8CQl;6JBy6ITst~yX0Kyugg=zPl z+*qidyZGC-N?6njGtro-P>#K>rhHH4)s+Gka4bJU$vmQdx0v&%-Hf=d__M#>b<9t> zW41;)WlV88$!Us5o$8N>;ZovB0*Q!ONp`JJu~jOCq&k}th?0!|^Y{xt zPe=(CsL}oa001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00U1+L_t(Y$GwxyZyQw< z#eeV3jK|YyaG=44Mr08w3BitHDFA=3K11WAwXH8A|bKkFNgrK zKwYBH;z|&rTBS~twEn1*iK$}8Ut`Z>;SZ&*-8LXN(&*{EdGDTk?ztEKV~AddbFZBH z7&zbekr27dKYscYKt1=G0Xzr#9C*9ntpdAxXvMpa;Ij*#ECLGv*Ahvt=Rz!;dk3Z3 zft%~%^0!2Wvy4xSb0SdYrP;d-KXV#sj(^seLwHpZgRS zCICu-rW;62XLNLwXfz5y#cSYJa||7;6JIx(7-{_0Ho3n7O``iObkoMPsx6}xhTlRr zEH*Ya7#th~AQp?EJ?1WZ2H=c+=cD!ZQ}7b@pv4=Fp#omsNM=*c>WaWbQ;kblYwI$k z>?kK1jWRPcgJ^YEdwnDVAq2%@5dgCOQV6cEC)vq5TwFTKPS)W@;wAv$aF|FW!u0esQp&yxdeZ`azaLL2cCrp%E}mg9 zFu=v7v!u2%NGY&w8(r5AHN+7-_~r5{<6{$ibKz}LsT8Gy0_9Q>&+|C{-pf?0Rg%dh z`FtK|wrM}~BshKQMFN2Ug+hT~FxZqTWv3L4Mg!M%v27awP1Eo^@8~8=DM=&}JPff| z>}VxAjLFGK48yoDfp*=SQ-4GS?I6Gd75w8lIE-*O+{w0`ardf}LiL&Wo&=##=%ICg zRNO=0dEV0JUo3z0Zvx)Fm3sH^Ui7)A1IK}Z)^{ARX%40aD8L2Ez+b>VaM0Q}TL1qa z{0-kuCedog!JGg903~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eN zIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)=GJFgh?W T*0Dx800000NkvXXu0mjfL&DnB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/bookmark_add.png b/Media/Themes/Umami/Icon/actions/bookmark_add.png new file mode 100644 index 0000000000000000000000000000000000000000..e2e8d311022d678d82efff324ceb848479779b1c GIT binary patch literal 1101 zcmV-T1hV^yP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00U1+L_t(Y$GwxyZyQw< z#eeV3jK|YyaG=44Mr08w3BitHDFA=3K11WAwXH8A|bKkFNgrK zKwYBH;z|&rTBS~twEn1*iK$}8Ut`Z>;SZ&*-8LXN(&*{EdGDTk?ztEKV~AddbFZBH z7&zbekr27dKYscYKt1=G0Xzr#9C*9ntpdAxXvMpa;Ij*#ECLGv*Ahvt=Rz!;dk3Z3 zft%~%^0!2Wvy4xSb0SdYrP;d-KXV#sj(^seLwHpZgRS zCICu-rW;62XLNLwXfz5y#cSYJa||7;6JIx(7-{_0Ho3n7O``iObkoMPsx6}xhTlRr zEH*Ya7#th~AQp?EJ?1WZ2H=c+=cD!ZQ}7b@pv4=Fp#omsNM=*c>WaWbQ;kblYwI$k z>?kK1jWRPcgJ^YEdwnDVAq2%@5dgCOQV6cEC)vq5TwFTKPS)W@;wAv$aF|FW!u0esQp&yxdeZ`azaLL2cCrp%E}mg9 zFu=v7v!u2%NGY&w8(r5AHN+7-_~r5{<6{$ibKz}LsT8Gy0_9Q>&+|C{-pf?0Rg%dh z`FtK|wrM}~BshKQMFN2Ug+hT~FxZqTWv3L4Mg!M%v27awP1Eo^@8~8=DM=&}JPff| z>}VxAjLFGK48yoDfp*=SQ-4GS?I6Gd75w8lIE-*O+{w0`ardf}LiL&Wo&=##=%ICg zRNO=0dEV0JUo3z0Zvx)Fm3sH^Ui7)A1IK}Z)^{ARX%40aD8L2Ez+b>VaM0Q}TL1qa z{0-kuCedog!JGg903~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eN zIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)=GJFgh?W T*0Dx800000NkvXXu0mjfL&DnB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/bookmarks_list_add.png b/Media/Themes/Umami/Icon/actions/bookmarks_list_add.png new file mode 100644 index 0000000000000000000000000000000000000000..e2e8d311022d678d82efff324ceb848479779b1c GIT binary patch literal 1101 zcmV-T1hV^yP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00U1+L_t(Y$GwxyZyQw< z#eeV3jK|YyaG=44Mr08w3BitHDFA=3K11WAwXH8A|bKkFNgrK zKwYBH;z|&rTBS~twEn1*iK$}8Ut`Z>;SZ&*-8LXN(&*{EdGDTk?ztEKV~AddbFZBH z7&zbekr27dKYscYKt1=G0Xzr#9C*9ntpdAxXvMpa;Ij*#ECLGv*Ahvt=Rz!;dk3Z3 zft%~%^0!2Wvy4xSb0SdYrP;d-KXV#sj(^seLwHpZgRS zCICu-rW;62XLNLwXfz5y#cSYJa||7;6JIx(7-{_0Ho3n7O``iObkoMPsx6}xhTlRr zEH*Ya7#th~AQp?EJ?1WZ2H=c+=cD!ZQ}7b@pv4=Fp#omsNM=*c>WaWbQ;kblYwI$k z>?kK1jWRPcgJ^YEdwnDVAq2%@5dgCOQV6cEC)vq5TwFTKPS)W@;wAv$aF|FW!u0esQp&yxdeZ`azaLL2cCrp%E}mg9 zFu=v7v!u2%NGY&w8(r5AHN+7-_~r5{<6{$ibKz}LsT8Gy0_9Q>&+|C{-pf?0Rg%dh z`FtK|wrM}~BshKQMFN2Ug+hT~FxZqTWv3L4Mg!M%v27awP1Eo^@8~8=DM=&}JPff| z>}VxAjLFGK48yoDfp*=SQ-4GS?I6Gd75w8lIE-*O+{w0`ardf}LiL&Wo&=##=%ICg zRNO=0dEV0JUo3z0Zvx)Fm3sH^Ui7)A1IK}Z)^{ARX%40aD8L2Ez+b>VaM0Q}TL1qa z{0-kuCedog!JGg903~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eN zIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)=GJFgh?W T*0Dx800000NkvXXu0mjfL&DnB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/bottom.png b/Media/Themes/Umami/Icon/actions/bottom.png new file mode 100644 index 0000000000000000000000000000000000000000..ecea1ee672ad6e5b4b07e8b0e716840f20eb5dd2 GIT binary patch literal 1111 zcmY*YYfO^|6uoHGQVNO_1)K~AIEIw81q*yMFt7+k=y)_dTLzSZygT@yz@U_>3k;a4 zTN(zm&==4`c|YYfBsw0Jmka^79~hPyHX8{0{>Koz<(0#*XH}Mbv*m45HLja1;0SK=FyaoW_V}N`j zz>@?3q!!>5ui;i06#$HRfiZai5co!yW-`11jD8LcX8HpFl81qJf#hNSr&m4(K{3<< z43%&2Z%Z%qjV~~?dvV`Sw=@nUDM0?d_KB)7#oPwS1g z&`LDn|BXd4m7}rZkyIoP!>BtP)_5dBj%6y0nJR3g%si?U7p=iXYjLsdCUJ7hxK{JH zPCQFt&FaLnbw}A2Fz6T%QYx-ID@dTY^;-FcQ+Ae9>jMHq-xpg$T*@3J5kUMrP zwbdzg()`M8^E8cfnud@zWuC5e$((*KtDT&qbIa)>@pP2Wf3V5zb49zTXb*x8y5&!O zoUi|&pv#-z{Yg=eZ_$9y7rp)^eE}tto+S(COZ(}i1D8sN{RI6%SBJu{-lJB{ek#^S z-CT(Jb|&e&>FkDSo>ZSD)$?R``SSG)#Z0MkR-l{{D3=SAKV_=s1gg0*)tpduPoUBZ z)Ovw>UZ9>YYhKH5SuAT=Dr+@R*t#s#tOzw5h1&TF?P_`3YI)m(^7f^Q_O|2oxtu~CTHI1z)$JQ0&8>)#7^~6T=146k&=gdV@=pejN;6Si7J1K0%LdQ((`MRS7{r^Iul33(X3@GvWZCSH81acy z!6r9uuT-z>5Rce#>@N0FJ#B_!s&}2n(!`Mx$^b<&JiA+sja$2Z!Q|=F_oZxUuN!Sm zeCEzsN#oYW)3^s74-K_!lUBCTyx zvrMY&IGlo;6e(4zo^kcqNZF%nzovh)s}fAo^VBI<<#p!m07ygKfA7yvk+{?TZ1BFaTLx!gpQ3uGdh zNFaI=h!iHt19828kSOPgM1)9`Jog~I6UfR*PD?2QGLb|g5Ge$bJCj64$nFS<{5J4w x3`+$7R)!4`IVmVVF_#N`KHnuRJ0p*i$m6=?+nku{R{Os%dP+b literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/centrejust.png b/Media/Themes/Umami/Icon/actions/centrejust.png new file mode 100644 index 0000000000000000000000000000000000000000..896dfa106be7b47a9277f420976407891a146bfc GIT binary patch literal 694 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s8-o-U3d9>=#%KJ9nNL8R@xx%5qAScq0VFK0X#brQy(dBoX~nOL{IeD4u!q`iBTU^Ci^}ARR8>s zG4qG>jd{zzUy!xxOO|sAxx8ij>(#T}GBy;Kd$)JJt8I^oRrQM36Q60Uu5{vgzf-c8 z+u9ihf(#5tp9wLZKd--1VcJ!uMIsGlzk`&r_ZRxTWZ2zmKF!5(CBr7OnagJKCPeBk zlywmhP*Hevlc8th0X3E*8Y~SQw*QwhzH?G)s{3EM+O{o`<(!h;`i>*bYj5Uo=y>K? zGtMdY571%jJQT!GemKFP#Uya`RfkuqyG1416jvx|R{z(IWlVl?chcEo8yJ+_j_iCR z*|J%>*DZSO$}oMW^-+tMPxAHBojmYy7_4tXcQps&8M^+Am*ciCe^rTP#mjic?_{6`bCY zVSFs-*bRTZ>6=AEXJ*c_(iZFbR$=pRmfwDHg2D?NY%?P UN}v7C2ILF|Pgg&ebxsLQ0H3x3jQ{`u literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/contact-new.png b/Media/Themes/Umami/Icon/actions/contact-new.png new file mode 100644 index 0000000000000000000000000000000000000000..9123074d35799adf58778e153d4cb96140c7b429 GIT binary patch literal 1032 zcmV+j1o!)iP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Rn1L_t(Y$L*C*Y!p=- z$3Jgoy4`M-&X$xSP5+pPB%*cs)1pM}Mu{Z;i_`<$3P-%^)euco)PvD@LFAt_1`-Pw z;|XkH;>AM~s-=rdH`_Kyi_6kthwjYmdp{3_q1!GT8V;KHOVX8$J^D1rW-Li!RFN9%S5=$4_&@+M*O5Z88{tKebtrWO&C=f9 zPHSr`0|Nu3dzx_{+Q_Q<->~#Ah$PWgn)5%0JE{T-S4Oeww=gsQ7x9J!Gt)VYF%${~ zGMNmCL;?{3!1FvtE*fs!XyB!umuYOY3CruTV=n+-P!*7Dc^tnuOS~b0gaPqYYca;q z-Q7)BR~I6JF$NK#udna+{5eDf&tLEpvAcN{fd13(GBh9KYX1*7fni=*{OLZ%$y4vJ z=dFX}a=GZ~jWKs$RS{yb7)?!0(eXA{se=AN7>!`F>P&a z6`^-jE|;TKEe0pHcY})iD#u$plB0&&D;Ua=kiexfb(YkvAdS8E^ zU9TNr&MQP-ePT71Wg{ZEu8W9Nge{&|15|tmlarIg<8>6T&y#f5alLShe104eVQg#+ zr4;#m9;FnvZ8JPF1i;AX1+>C3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgF)J`IIxsNSu|_!n0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0Q>;y08cd6u?+wK010qNS#tmY4c7nw4c7reD4Tcy00LD>L_t(Y$F-EbYZOrw z#eetCY$OSOkWDf!f|Z4cl^`a9|A0*jD?72Z7C}rSDt`PE7B*rb+lwLy>V`B1wKChp zC2=8$tn2PhGS_0>d~7mnva4PgX67>QoOjN7ci=yl{0oq+jRgTpKc@5W-JH?U#gB#+YF9+vh8W z_iAfn;i(@~yMEEadcBV4d6*d@f=Dbh{|rV+8cgU5ywW%}b_wyOI5hnli=JH;`@Rf( z)WTpKPy%LVV1}7tkzod~jYs z%K!n%XCR=0Fowx7LdtA&it))3wUr;7IobFC{2HiWX}Q)PYEDghk*M-8oy5H3$k=IG z?Gk=aZ71_kNO11#9FwKDis2LlL9nND`>j5LD2nbpcy#a9-vEAHYh3Mf z;7{(1$YCR{8?Al=KB$&GFYmhv0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>% zI65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(b dH##vfD=;uRFfi7!MmYcg002ovPDHLkV1iNAT0j5* literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/document-open.png b/Media/Themes/Umami/Icon/actions/document-open.png new file mode 100644 index 0000000000000000000000000000000000000000..9e4a9780c66fb64eb859c69c33b27db420c41580 GIT binary patch literal 1093 zcmV-L1iJf)P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00SaPL_t(Y$F-JCXk1kk z$A9;|H#13Wqo7&XfD|PK5nQ+^twpSY0Y%){DQ;5TxT|ZI7FQy!-1xC6qO+Q{U=)f% zsc^7zf z9Kb}xZES2HBB&~=ikab@qgX7m_{<_^){6q5R;wW*JbwC&`{vuP&yNLo?@d*mi1fnW zYXDIcvHbaGc<+18Pn~_5EX#V?RMlW?4n%Sj$?@Kk<@*2;0T+S`SzLUs@9hshqo{;g zpl4sXJoV-AwTw%bv=5MFIe9LuZVS~PgdhmIMOMJ>et{R?yt1G4fja1N(3xjv!8K3* zd7^e5I|k1Ba^SsZ>$dX5W5;>^%!!_3r|ubR2dCmp!Z+T#G6&9vav&nasWCTQVyltx z$4;9`Y^o4)-_Zg=7&OOK5v?2^ysHETQ4mZ7P=hq{1TJU{0rI?)Ahvgn6o6yE+~BrNXbS8CzXJ*E-%}005UK#FfA}SEig7zF*rIi zH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-Lg zF)J`IIxsNSu|_!n000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs00000 LNkvXXu0mjf@Ui9R literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/document-print-preview.png b/Media/Themes/Umami/Icon/actions/document-print-preview.png new file mode 100644 index 0000000000000000000000000000000000000000..760d4c62798d9e72616d6b121c0b7289b059d187 GIT binary patch literal 1291 zcmV+m1@!ufP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00a$5L_t(Y$JLcvY?MV5 z$A9y++jhI#i>4R4BwVCzjH0QOLIRe?@Sv?cXhGCc#h941S*b+~QI?jpm`F|Z!50h> zE7TC9(2LlLAvLy@fTmmoxfEzsXzh(lPv_>5kdTNF0-Mc7a&mHLHl~5JnZ$QgY|lD% zx}D-;JMCSgd~~#rv)w*QOG;_{_An8V5!h@tG)?y1a!&uK`+taf^&VaG&76Zy00ds zsI1(Pc}IYhl$5J|ef`W&il(c_&B=~_nt$o$(Rmi4@DddvaJz@dTDoj%4cg1f9feFe z`jd8)l{*ThRH>&NKZ=x zs3|LVYzWm23l#@yDt7FouKo}vAz7ThkmFq=oWIn`khHS)k#LT;9wAH%z{i8pk~Gt4Y^%3G&Eoe4`=IJ zZ*$?&uN*#DgRirh>OCKlm%E0>rf*o1@fZmSiLA?CPu2Uo7#QdW6#MsAGdS2ksiu=U z`FeK`Jv}{GtybFGPqN8g!p)mb+K#odp`eh5A6ksx@29b;k&Q3Eg2iG5AT}ljzdwM> z<-+gRX9ox&P)f10vXX%A$1o&PN({rG1oVKPQpFpMgvsv)G4J1=mBndJb1mMXkh{&1VW5wS<^HkBTUo!U=l#rbrcG# z)k0)cBx7S^pumuFJaWSzKK_1`QYdL4l^kCyg_IJd1U{cHw7xPF@Nx6^M~j|%GLIwQ zHE}QYz`|5IyRP8#`I@Ey{DHuy5hhd7@UWYlRZk#%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vfD=;uRFfi7!MmYcg002ovPDHLkV1i-P BOG5wv literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/document-print.png b/Media/Themes/Umami/Icon/actions/document-print.png new file mode 100644 index 0000000000000000000000000000000000000000..0c35a115626b7974612b3ef4eb2d47b3d15c8f1b GIT binary patch literal 1030 zcmV+h1o``kP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Rg~L_t(Y$JLe3Yg|S)8F6E;^IAWxg6PSmc_+;IF5tXdRMGH5!|@(EkO|A`+nk6 zQRGMNkq4jfFt z(-Ni6>_|{57e4~-r!#OI2gh+ZapGl$hlkNxC!VtV@!9UW#T%|qeEJyxjYb0iYc0kY z1_z(#`0+t}-^cSjHa0erJtKtz!1YqOcrlHuwgX>Zt1&x2&)n=R)|wsf=$PYBC=__@ z^;0~2_!qanzXkB|rHRj{9+yTQndf=(q~^M=xUMV9%ga(K7n6SIsG_xcjrqGjbMoY? z96EHErKKfCM@JEnL~mnEl9gCN=}pNeC%<8A?3IqdI|%+pT$-35j9M6DH$ep?o&+K`Ms8=(lkLG6 zOD>lqigqBV*XxXo6gV=_PkpV91|+Tn5h091{%JI5Ha+w{g;ENo+R|yQ>FvpEDS{4y zD2fn-e11Q@*pO^uAl6`|ouP@5{uayQ8g3<>D1!{Hemm`hdg0QQ$e?+1PEo#{vz& z1^xi4Koc#&X1Fth*w001R)MObuX zVRU6WV{&C-bY%cCFfuSLFgYzSHdHYDHjtgI}7Kp+$fMIw<{ zEG{T0kVqslnM^L1E0s!>N~Nw))zs89Ha2QBnx>{EtyZhk>2z({);3*xd%Iq*@9600 z?Ck97>gw(7?eFiWC~9C}z+f;84h|ZP#-X7hlgTtZJUlWoGCDdsHa0dsK0YxqVK$p5 zCnu+-rlzN-XJ%$*XJ_Z;=H}<;Ef&kd!h+RmU0htW*=$QoOLn{6;czT3FR!eutgfzZ zZf+iURG$pMOPZE)5l*hOz21idgTo`(oHKFp*`=k*RQ30E08ZKG5@NZvqLN?$V@a`` zbnioMAGj&|`Cjgn?e|++sBo@Uob4yMyV2Ggl*w~`xhKr8s&=S&d#cC8Y4EOyW_^L~ z4ElO~{1N|33*~1$*~3@3mxuPf_SbN#GZk;y${#EFJ3!aI(aJ}P>^5h*Phnlan%-z} z^ZHtS?axM}!`D~f=6ST{`}{i5Od3u zE5e72&>nmD3$}~T%|!99K#CpG^SdOB?b49;_-x*8is#Pubp6r>g)BhR5`0m7_4Bx` zxY7wwY<~I}-BI~q-$ww)q;Z#|xqN9JE9ZJ1Uz!Id!6X<&6oUv)CnH&5r&wh834&k| xM4c^?{3p0pm@CS^1tvj~3?iICMx>KW7BhlHlH0+6K0az|QBqt=Y-q*$ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/document-save-as.png b/Media/Themes/Umami/Icon/actions/document-save-as.png new file mode 100644 index 0000000000000000000000000000000000000000..00fdfe01ae117e41de48a02b22b1b34a738613b5 GIT binary patch literal 1048 zcmY+8eK6Yx9LB%7S*M)4X}7D|>^2>>wYFb$L!w=TsFyX)OBCZYUeX}(YnsMO5F)fB zUNVX?u_D)v-Wv3?0stqgY+nI@hh5_e0I;DG$wYu&vHgqWR{`MN)Q~7LK)yQwzYM?v0DL1r zHUh9h1;94}9AHU)j64nids!jz7XTa{T7?PlXvqV>R7ISk1OWi2l-w+3qAW&3`}vsI zXtb1qvKV=liOiB%Yh*F72^uGPAp;bGg~YB}Jtd`8DEd@xbr(jkPVU&Aq)na=BcgP$<;5$Hqp- z?%z==l`7S^TCJXVFg7_eHZwb;v7*&zwOWlv`_gqfoo+$5u%KI9T-59JOG`@zgTZJt zE-xEbOe-dnX?1mVZEbCReSLFtW7Eo(*=)8vx9slj;%*MA0KkV66&?-t2N4A4M+yl& zeL6mYo^c_kv`j2%ZI|`l=)XCnn4FrMoBMNZYis+-UuMhSJ3Bl7yhJ*;PY(c?l+S}n zr7!wa695=eD2Yh%Q1K?7M&Cl5___~KiTtUiVBfW?nf}-03wsWSf6wofN5!rZ&^mS; ziC1jY4}Ukb!Hy+9>lWqso6r#pR%JA;9GsndJTUO)`>F)XYnNOuYpNT%G>ONirhWAG zxYk4Oh#5VH#%;!9mY>A0HkHVp-woS8Rqs0M&hOL5=X`{1OK#X#M#`IPa5(@}8zWB$aE^4Uc{%W%ih-zL@iZ=_nuBj2_Hy8|7rIJbCq z*O}UC{q*xlgj@?(xQORA444CNOzgc2ef}ldGuXo|M#W{z06J_nO~+i!&%Q?NyWA zh+%uGf})*iUzQhru)T4{V1K@$!uj#X=llV{jT02dNlWI?@TqKCGKU6Uh!^6C;5-p; z3f32Y>{C3}`v`*I5u|b37yG|L*15Edvs~~(uvkyT+Y{?U!Fu7neDGN8e}xCVCvgA( Mp+Vs!SpZe=FLHDF)c^nh literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/document-save.png b/Media/Themes/Umami/Icon/actions/document-save.png new file mode 100644 index 0000000000000000000000000000000000000000..747686b650a599fcb879f9faa0468de8574a7100 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/edit-copy.png b/Media/Themes/Umami/Icon/actions/edit-copy.png new file mode 100644 index 0000000000000000000000000000000000000000..94c72fec54fb6c3bb50fd7a13d797816e8ab31a2 GIT binary patch literal 891 zcmV->1BCpEP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 T*0Dx800000NkvXXu0mjfXQYbf literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/edit-redo.png b/Media/Themes/Umami/Icon/actions/edit-redo.png new file mode 100644 index 0000000000000000000000000000000000000000..ab73631109595c31b33c4d71ca04bb3fe46740e7 GIT binary patch literal 960 zcmV;x13&zUP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 bF)}MKFgh?W=UH5&00000NkvXXu0mjfY-sKY literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/editclear.png b/Media/Themes/Umami/Icon/actions/editclear.png new file mode 100644 index 0000000000000000000000000000000000000000..fa8cadb5726aacd5c9d48ee4062c1cd72e5d0145 GIT binary patch literal 1381 zcmV-r1)BPaP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm0000;0000;07l7cJ^%m!24YJ`L;(K){{a7>y{D4^00crwL_t(Y$Gwz!Oq*pK z$Dh7!xdg^I#^E?biUC{#LUckzNnpAqGsA!r78lWpOlGDmB)YAvfWzs=WCNFQ6sAsO zW|K@HPNH&+%DHi0r5r1CRA7|5U1`5Q?|kDQW(o`Fn|%Lyp5O2H`Cacr$p7(u9>Z*+ zI&vUbzV6H*(niP zFNC7GC^l1i%Jsr`b20_Ct->20h=sg_2~^A_W{ONFlo~rkluez*I_#sP# zlCut&8x!Mtk?&moaT2u=+Z2_(-(;!yq%F$xTrsVP$D%R`^-93-RG9Hm^R5qD*@85cr=pOkht0l1nkv*^&9DdOoG86_?7z`7U&`+m9J@5hYC5(1e0L>z^p)A-i*0GZU81z>QplsSq z0p6JD^@i?t2o^`8Fxl=8{rHO%um=XsS=5&XYw7T%tG0+{5d8J07V~QPVaV$&&{xGk zal?uZS%j(XKxlgXU{Jq``QccMw`|9~=}dIjMC*&v_yg3}e6{!t)gyT+q@|;3r*ZVw zvwv1mWdUWQ17;OFFmclt^ZkJ^+&zwIc{1v+_}`PJ3c50rh~sL3t*M-;y!P!PF_%8G zU7i?t4~pNMFx=>Y{$H#x+37`}kT)!kh<-!N`{XhTy0AqSkqHJBRMgRZ+C3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgF)J`IIxsNSu|_!n000?uMObuGZ)S9N nVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjfVDxHP literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/editcopy.png b/Media/Themes/Umami/Icon/actions/editcopy.png new file mode 100644 index 0000000000000000000000000000000000000000..94c72fec54fb6c3bb50fd7a13d797816e8ab31a2 GIT binary patch literal 891 zcmV->1BCpEP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 T*0Dx800000NkvXXu0mjfXQYbf literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/exit.png b/Media/Themes/Umami/Icon/actions/exit.png new file mode 100644 index 0000000000000000000000000000000000000000..b757c74afcb372ea4afba5223af7936d5034dc88 GIT binary patch literal 1264 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/fileopen.png b/Media/Themes/Umami/Icon/actions/fileopen.png new file mode 100644 index 0000000000000000000000000000000000000000..9e4a9780c66fb64eb859c69c33b27db420c41580 GIT binary patch literal 1093 zcmV-L1iJf)P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00SaPL_t(Y$F-JCXk1kk z$A9;|H#13Wqo7&XfD|PK5nQ+^twpSY0Y%){DQ;5TxT|ZI7FQy!-1xC6qO+Q{U=)f% zsc^7zf z9Kb}xZES2HBB&~=ikab@qgX7m_{<_^){6q5R;wW*JbwC&`{vuP&yNLo?@d*mi1fnW zYXDIcvHbaGc<+18Pn~_5EX#V?RMlW?4n%Sj$?@Kk<@*2;0T+S`SzLUs@9hshqo{;g zpl4sXJoV-AwTw%bv=5MFIe9LuZVS~PgdhmIMOMJ>et{R?yt1G4fja1N(3xjv!8K3* zd7^e5I|k1Ba^SsZ>$dX5W5;>^%!!_3r|ubR2dCmp!Z+T#G6&9vav&nasWCTQVyltx z$4;9`Y^o4)-_Zg=7&OOK5v?2^ysHETQ4mZ7P=hq{1TJU{0rI?)Ahvgn6o6yE+~BrNXbS8CzXJ*E-%}005UK#FfA}SEig7zF*rIi zH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-Lg zF)J`IIxsNSu|_!n000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs00000 LNkvXXu0mjf@Ui9R literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/fileprint.png b/Media/Themes/Umami/Icon/actions/fileprint.png new file mode 100644 index 0000000000000000000000000000000000000000..0c35a115626b7974612b3ef4eb2d47b3d15c8f1b GIT binary patch literal 1030 zcmV+h1o``kP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Rg~L_t(Y$JLe3Yg|S)8F6E;^IAWxg6PSmc_+;IF5tXdRMGH5!|@(EkO|A`+nk6 zQRGMNkq4jfFt z(-Ni6>_|{57e4~-r!#OI2gh+ZapGl$hlkNxC!VtV@!9UW#T%|qeEJyxjYb0iYc0kY z1_z(#`0+t}-^cSjHa0erJtKtz!1YqOcrlHuwgX>Zt1&x2&)n=R)|wsf=$PYBC=__@ z^;0~2_!qanzXkB|rHRj{9+yTQndf=(q~^M=xUMV9%ga(K7n6SIsG_xcjrqGjbMoY? z96EHErKKfCM@JEnL~mnEl9gCN=}pNeC%<8A?3IqdI|%+pT$-35j9M6DH$ep?o&+K`Ms8=(lkLG6 zOD>lqigqBV*XxXo6gV=_PkpV91|+Tn5h091{%JI5Ha+w{g;ENo+R|yQ>FvpEDS{4y zD2fn-e11Q@*pO^uAl6`|ouP@5{uayQ8g3<>D1!{Hemm`hdg0QQ$e?+1PEo#{vz& z1^xi4Koc#&X1Fth*w001R)MObuX zVRU6WV{&C-bY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00a$5L_t(Y$JLcvY?MV5 z$A9y++jhI#i>4R4BwVCzjH0QOLIRe?@Sv?cXhGCc#h941S*b+~QI?jpm`F|Z!50h> zE7TC9(2LlLAvLy@fTmmoxfEzsXzh(lPv_>5kdTNF0-Mc7a&mHLHl~5JnZ$QgY|lD% zx}D-;JMCSgd~~#rv)w*QOG;_{_An8V5!h@tG)?y1a!&uK`+taf^&VaG&76Zy00ds zsI1(Pc}IYhl$5J|ef`W&il(c_&B=~_nt$o$(Rmi4@DddvaJz@dTDoj%4cg1f9feFe z`jd8)l{*ThRH>&NKZ=x zs3|LVYzWm23l#@yDt7FouKo}vAz7ThkmFq=oWIn`khHS)k#LT;9wAH%z{i8pk~Gt4Y^%3G&Eoe4`=IJ zZ*$?&uN*#DgRirh>OCKlm%E0>rf*o1@fZmSiLA?CPu2Uo7#QdW6#MsAGdS2ksiu=U z`FeK`Jv}{GtybFGPqN8g!p)mb+K#odp`eh5A6ksx@29b;k&Q3Eg2iG5AT}ljzdwM> z<-+gRX9ox&P)f10vXX%A$1o&PN({rG1oVKPQpFpMgvsv)G4J1=mBndJb1mMXkh{&1VW5wS<^HkBTUo!U=l#rbrcG# z)k0)cBx7S^pumuFJaWSzKK_1`QYdL4l^kCyg_IJd1U{cHw7xPF@Nx6^M~j|%GLIwQ zHE}QYz`|5IyRP8#`I@Ey{DHuy5hhd7@UWYlRZk#%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vfD=;uRFfi7!MmYcg002ovPDHLkV1i-P BOG5wv literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/filesave.png b/Media/Themes/Umami/Icon/actions/filesave.png new file mode 100644 index 0000000000000000000000000000000000000000..747686b650a599fcb879f9faa0468de8574a7100 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzayiiP3MF0Q*bc0Qqqik+)Yn7pKRaaAPgH&#CYj}%X zY=Ku*S5t0qYkG}jYJXa8aBFyoWp|5gZg6XEaBF;lZEkRDW@uz=g?C_mZg7f%XK7?x zdU9@XYhHYLZH%-!d5Tkcib-yQO?i<&Y=%N^h+>wzL2QbFt8WmiLz0PwNs3>RC0S*lD%7&!CaTaUYW*Tn#W+A%3z(#VxP`qpw45V&}XC4Xo-_+ zrPFS3YjSmKbg0yId2Ds5)pvPye}R63g@BHajgPk5k&%&t=*v!_}*4Nk9*xK0J z+uPyc;o;)p;^N`r+9?6?CkCB?eX#P^Yioe_V)k( z|F$mNT>t<8H*``?QveGYFfchgL`X?iR#{tTX>)digpZt^p{}vAwYj~@)70JF-Q?}@ z@$>Zc_WJw%{r&#_{{H^{qumna00001VoOIvli>s3wg3PDQAtEWR47w@%~eywKoo}I z#ih8ryZdP&6xbGm2X}Xh6)O~nQX1q0*k8=-W|%4Z<{~@ix%deAuY`O6fZ~k6Z+HK) zqLQkK;qKhL@UQor@``F>WA|WxQ)!4dq@`ztk>J9bjt)y@fsYqtC;1XWqKvK7(wUj) z1u3zLq9{W^DW!dF{2i@BVr>V+vls!&qX+TQBumYbS+bf=5H&I4HzFpEK$fItD;yy*ZIq7~ z#x<3(S1#htn5vEQk(jlH=O>jhb_KHOiZNAn*RRA8LXywlRlPkp^2>>wYFb$L!w=TsFyX)OBCZYUeX}(YnsMO5F)fB zUNVX?u_D)v-Wv3?0stqgY+nI@hh5_e0I;DG$wYu&vHgqWR{`MN)Q~7LK)yQwzYM?v0DL1r zHUh9h1;94}9AHU)j64nids!jz7XTa{T7?PlXvqV>R7ISk1OWi2l-w+3qAW&3`}vsI zXtb1qvKV=liOiB%Yh*F72^uGPAp;bGg~YB}Jtd`8DEd@xbr(jkPVU&Aq)na=BcgP$<;5$Hqp- z?%z==l`7S^TCJXVFg7_eHZwb;v7*&zwOWlv`_gqfoo+$5u%KI9T-59JOG`@zgTZJt zE-xEbOe-dnX?1mVZEbCReSLFtW7Eo(*=)8vx9slj;%*MA0KkV66&?-t2N4A4M+yl& zeL6mYo^c_kv`j2%ZI|`l=)XCnn4FrMoBMNZYis+-UuMhSJ3Bl7yhJ*;PY(c?l+S}n zr7!wa695=eD2Yh%Q1K?7M&Cl5___~KiTtUiVBfW?nf}-03wsWSf6wofN5!rZ&^mS; ziC1jY4}Ukb!Hy+9>lWqso6r#pR%JA;9GsndJTUO)`>F)XYnNOuYpNT%G>ONirhWAG zxYk4Oh#5VH#%;!9mY>A0HkHVp-woS8Rqs0M&hOL5=X`{1OK#X#M#`IPa5(@}8zWB$aE^4Uc{%W%ih-zL@iZ=_nuBj2_Hy8|7rIJbCq z*O}UC{q*xlgj@?(xQORA444CNOzgc2ef}ldGuXo|M#W{z06J_nO~+i!&%Q?NyWA zh+%uGf})*iUzQhru)T4{V1K@$!uj#X=llV{jT02dNlWI?@TqKCGKU6Uh!^6C;5-p; z3f32Y>{C3}`v`*I5u|b37yG|L*15Edvs~~(uvkyT+Y{?U!Fu7neDGN8e}xCVCvgA( Mp+Vs!SpZe=FLHDF)c^nh literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/find.png b/Media/Themes/Umami/Icon/actions/find.png new file mode 100644 index 0000000000000000000000000000000000000000..3f3faf3aa094fe54c7f84894591a1cb73b93aa5b GIT binary patch literal 1254 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/folder_new.png b/Media/Themes/Umami/Icon/actions/folder_new.png new file mode 100644 index 0000000000000000000000000000000000000000..02cb4c962d07e203c588e54990020e5bc13152b4 GIT binary patch literal 1014 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00Q_)L_t(Y$L*BOYgAVl zho5uqnV6Y6@vCNt1uH}l;$+cAkP6cN2ku1Z%C@eGSl5E!KVVi;s&v;)5EpTyD;2U3 zS~M{+VJ7hd(IzvQWHQNo-MJs9`{e-<(K~{&2dI1A2f-U{Ynq8vrUEU})$lQaW7$ z2M&C;sqHw1{ynQC^u$HaIaWgSDW&tTgf5H*#ZzM{Gv4X|JkMipZkBuVhN(ZdQ(eR1 z%sStm%=6`uF9_6`{f^p8y*aUGxp@2XV5aa0nBAlxn;qur{b??J_cg8`{$qGaz^AF_ z4DK$FN)F(teFz~@PLfKqXAs!+PKlmAodIC2?orxkz2|gjYAMAhau-r8kai|WB#sdH zk9kq(VdN0yb|ThX*Cm@B-cD_7fh23q{md?pkS`siwDOQjd5kx!lDks}0T#O|*sg^} zvC`O7+3YYQx!;kFq~+^&EI|jfH6ZpG zN!^>46jwHcbW0Cx@?Hc;0L;wg=-b^-q0+*&3Zl3jNc?H&WROlOTE3=KsfKFx4`~0F zI)D^F@4yhBryX)jbp)W}8`1f;rUL^>uE(1|)%61ZYlA&zyVt z+tiJJd5A632XpO5nJHcjZt<_ zWGsO=Kx}zyJEnmk=GsF4Hq)$ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/format-indent-less.png b/Media/Themes/Umami/Icon/actions/format-indent-less.png new file mode 100644 index 0000000000000000000000000000000000000000..139ad7036d70a961e0e38effa660777b5e384ef3 GIT binary patch literal 840 zcmV-O1GoH%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00K%$L_t(Y$F)?;Zqq;% zJ$4?8q)`eAi2)}P{T#jk)`-vskPs5cG_Z-Gim0fl(t!8^*dy^jPNXd80u_Zc4apo9 z^RV4GjThlsmd7*pIrrRiXMq14mL42^Jp4MF&F*aiG@s9Zd;aXj{n%vp{X1?d4nK6M zakH1%FQ(IoE`kD}xRoB%!v{^xAxlS{-5LU+95W_Vro2z6eS`BU=e%9OzA-|4A5}vW-jW7>9B-^>}3Mgng(28 zFbv@P9hEsjiip`tjjUga5-I7r?SkO|zTb&gAX1)KtEMTDFh3E&U;zJZ$E-ctYob^z z!f_lpH3!vN6;9PbtyV>)awA=v>dpv;aqvWy>M2awQgo3758(Sgt_)KyhTf$$x#2eO zru}*?dbG==4@^TKKXlzX+OOIx;Z42_4~lgAK=(8Wx9(!!+s8ro007YMokTL#C#&>A zFqvRUf!Y4*2iLs~&-22PL~pGaD-NPW1?JR}8x8>6TQ0np9$?gIt)UknRg147*xtUL zm`SBlMVhcIEB0;(a=G01Ce-`TK8 z02)&bUo+vKmH+?%C3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uR zFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGAl4JIxsNj SSzM$50000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00LM^L_t(Y$F)?=ZW2)# zJ$Lw7AcU$kK}HsKTc5-y&{bn=vGFG{rcm|`IB8>}F>0+3pnDVF$AJLSg{h&GB7Dci z%)K*|5m?}zOmb)L%sJmV=erm9&qL|S(VN3B!{PA$GC-5bA((7!-}LeN_Q#9tu+dhfE?>l1iW1O}W27Nh_GJ{-S~wB*rd$)n`- z+7tmp;Lz22#bc-R1Y}S#AP}vj^SFzX9gBqX}spJ=z+ zXm`B3)9#?t>7aMsGvC!;)lschQM8Njdp5F7BNQ-XkM8&A;W!Q&jRr&rQX)6nbOz>F z5ChJCY$5<~4jeR_&Cq^XAY>pYoU| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s8-o-U3d9>=#%KJ9nNL8R@xx%5qAScq0VFK0X#brQy(dBoX~nOL{IeD4u!q`iBTU^Ci^}ARR8>s zG4qG>jd{zzUy!xxOO|sAxx8ij>(#T}GBy;Kd$)JJt8I^oRrQM36Q60Uu5{vgzf-c8 z+u9ihf(#5tp9wLZKd--1VcJ!uMIsGlzk`&r_ZRxTWZ2zmKF!5(CBr7OnagJKCPeBk zlywmhP*Hevlc8th0X3E*8Y~SQw*QwhzH?G)s{3EM+O{o`<(!h;`i>*bYj5Uo=y>K? zGtMdY571%jJQT!GemKFP#Uya`RfkuqyG1416jvx|R{z(IWlVl?chcEo8yJ+_j_iCR z*|J%>*DZSO$}oMW^-+tMPxAHBojmYy7_4tXcQps&8M^+Am*ciCe^rTP#mjic?_{6`bCY zVSFs-*bRTZ>6=AEXJ*c_(iZFbR$=pRmfwDHg2D?NY%?P UN}v7C2ILF|Pgg&ebxsLQ0H3x3jQ{`u literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/format-justify-fill.png b/Media/Themes/Umami/Icon/actions/format-justify-fill.png new file mode 100644 index 0000000000000000000000000000000000000000..b009ee3b3cb4edd08d8f5440e80ffa7236b4d18d GIT binary patch literal 683 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s6sPZ!4!kKLiF_Uv2{WI0`V#dUp^-QVIFTqX&hVB9^arE{BB;Lez7$}7UOv!6zk zyjee4rg2#&V}iMWz+n}JGaT!xBkI^!35jyGeq}ksvo$I-Xwn18i7hTJCiR-TkZfhK&LeuZ8|^XA@-R-BG?$)}Ul}tZHY` zPUo4N99>5zFP&6T#A5ybG0W;$A-1*?pS_w&tk$aZ*PCx-X-nK=YqPEIU+X@F%K6XZ zCY_agG4t=4xc`gudEJ#?A1_?L+w$X@$=PeX zwHdRH8BF!Lxi{{8vB3{1=R5j&mp^ig?Yf=7@?Gd20|SFRdP{kVo554 zk%5tcu7Rblfmw*5g_W_Xm7$rofq|8Q!L-j7hA0|x^HVa@DjAFn3=MS+%ybRSLkx|q zjLfYJjUXDHN5(8+U|^60*$|wcR#Ki=l*&+EUaps!mtCBkSdglhUz9%kosAR&0|SGn LtDnm{r-UW|CRhRb literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/format-justify-left.png b/Media/Themes/Umami/Icon/actions/format-justify-left.png new file mode 100644 index 0000000000000000000000000000000000000000..224d70023345320156b7f57b56361d1cd02092be GIT binary patch literal 684 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s7XPZ!4!kKxye%Yt8=p+xlL(soX1_(z6=1HgdUd=gd@`Y8CYD zjPBN?LuVKmWU9A|T*%t1!W6Q4D#Ieq6<61p#GcnvVoJNf{(|$oQY1?z^CXuiyPjUy zYqlV0XN+WfiTr(U1_K^v!Fi%gNg_-RB?mgZmIf*8s?d{WIJ2fF)7Vq%pcm`u|i_Tee<#d z7haZZc{Jm`qx=D%0G+rsoh?dD(Ip$|ydE`mGTh7&E7NodndjV)8Y%C8{_N+uSDsHe zU!W`g^Uutg)@SE_*;jQg!J_0}&Hb4^YUfJ4G8bje>gb=Td6-8`sZa2tuv3LoNBm&} zo6AP*+j4o0UiMsnTekoA-}!<%vv1d4w@(q{O@E}6@REUnLAAs+q9i4;B-JXpC^fMp zmBGls$UxV?QrEyN#L&XZ*wo6| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s8do-U3d9>=$ap7uNBAhNeSqkE;mVZP~~ z%(GrF2_?0j(3cRlKiJ}-BDJ5Pxly&VbN&%0F}C2!kR8`qHA|v2WY?HyTFp&6xAXaz zGa?V>bMJM0z42yg>~|&3fWrn==GV7xGr75A|KlaAeqKo659i)=KJD^e>-02_<@`%5 zbN$x{7#K4!wBK_T+Hzap%jN2;rBaT2>$Wb={?F>LWQBrMMW*KjkC_ffmw7E+*toX(+sd#21r{&OkMm7C zG;Z$LbNmRmaA~^EPU?K6r;CYTW8|ddp{YO zVz1wyH!nK9-YI_ap7qD>EeNo<`04o1#Sf2PknvZ0*8h6e=bv|CbTpi|oDle^Xq-CJ z(fx>|l9dyOJIBYlUQ1e*kj)2?C{7#J8- zOI#yLQW8s2t&)pU6H8JVj0}tnbPX(Z4a`CeEv$@9tqjey4GgRd45odyFhtRio1c=I zR>@#wU}&goV5Vzm9%5)@Wn^w;Xav#lJThhp0|SF3$cEtjw370~qEv?R@^Zb*yzJuS c#DY}4{G#;P?`)(P7#J8lUHx3vIVCg!0Oe{Bs{jB1 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/format-text-bold.png b/Media/Themes/Umami/Icon/actions/format-text-bold.png new file mode 100644 index 0000000000000000000000000000000000000000..87bfdb2c7ca50dcd52f718f6bc29454679204007 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00S~fL_t(Y$L*9)Xk0}c z$GA{;!VXIOb-D?q0rJ>X>P3` z(yGvl^wwPb0}_grgl0|Hy4k#auibt7=k3ee-FY+Pq1%!_ZfL#ep*_?(I2BX@yG+F3i1O*liAVw9gNQ2QtL+E{OZFK(SpseYSi#oz9jqN-6Yx-K!3!Kb60<|wBEkN z{Ok#@{UY1Q*7dOi$w;6l+=-0oAgh<~+wU2YGs@7dI_$bcPc#&;b$x92Q)r`rLTBE2 z^-yrdsG-gTmNTXe;JZJwrp-Ko)x3kgL@cOMI`h9cz+Dbxhjff2X<|P z`O5y&j_tB?bpz#E6V#^xk{>*F=1S_R01743Z@>0}rkCodI389rd6#?a;xE@vxtz0$ zYdOESv1lX!$mu5rj7(gP3*ntSKAfa?G8H&=2IGE_Ky6+E0I1DN4~iCX8w|{o z=nh8}Da6T|R!rWv502byZ3q*<=rul#~Pe6d8 zGZd1dNvHk^u*IgMqqC|#&}V-1!5h(>#h_btEG{iOg+ej7ec+yO#OX~wALLxZbvKaD zrmb++NX*aAZn5l^hgR&5jqTgp?FUh@ZdDQVs~CUlm7wNR5b!C`{1lpBLC3=g{_yiU z{}3=i!OEw9(mWD;(p3A5Y)7ZRS3;9P4 zMG{@i$jr0_42_>pDU=L$hV~$P-$kWb#nPSIo{+NbcR~p9dIk=wel3WOP#BgJgG1xz zQ%m1|a(f3rrOI^jxuj;48fbU|xwWjzh5T~y+Q(nC1H5$V!r7{|{#m^LPzONJ73tL~ zmOc&Oy&VAIv6s{7yGlBJw{`o38R=Ws03-k?t$1V#34rGpfBe4p#Wz27N@ho^T_KbX zxBAH)F52d3^iZ0hzmTT8pQsAOHXWC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfd};CU literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/format-text-italic.png b/Media/Themes/Umami/Icon/actions/format-text-italic.png new file mode 100644 index 0000000000000000000000000000000000000000..cded0616b95a19998054647d36e725b66440a1cb GIT binary patch literal 952 zcmV;p14sOcP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 Hl+XkK3f^zp literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gnome-stock-mail-fwd.png b/Media/Themes/Umami/Icon/actions/gnome-stock-mail-fwd.png new file mode 100644 index 0000000000000000000000000000000000000000..33d884326cc95c6dc1ddcd46c470080420356fd0 GIT binary patch literal 985 zcmY+DeK6a19DqM6o8aOyZPZ!4ZADMV{eJPf*o*99xT$KqRrw*w zSJa`T6KySAZL62n=FFCan|2cPlw(G^_CkY3yhid{zdN^ow&(8o^ZE0sOiGNUIC(e$ z03|+-#RdTPeaZF!Fw?1h3;?8D<_RW1_YG%)`vw5`cZ> zIRK0%fWw9Ifm0saGi z_{~jqsc(f8q~mxcOb(GLme;Yrkg3jLUl2D#{$e55W$>Jl_uibgBZqwDB#HPSVrDJ7cj zLbKgywinIyp@l)TFod3upy$6KgsK81$pG+{n4Opm2OOM2Si&o1SEY@&pDi0#xNhk` z1AtxjXR)knMBn%m0K4PyEN05F@f?mKP+!M%kW(FR3B*43wKZuss+=h%S=B*sTaCQ<0?X1VwzqG z475XO^@kc~^4!C&M;|SZZc{YXd3d_Iw#pk)+b7>6)#|CF16=96V%cJ`)EpI2n4f410-r1910ADdLD>62Sa=E^$p{H!;NizHjDcYdo00hcg0mu? pP;e3G7>);F;X!yr3QlLxBN#aS-(V>t>)!rR@iB?4?>^yN{Rb2{*FyjR literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gnome-stock-mail-new.png b/Media/Themes/Umami/Icon/actions/gnome-stock-mail-new.png new file mode 100644 index 0000000000000000000000000000000000000000..f810f7c44a7954fdb654d30952cc9743cf6fffc6 GIT binary patch literal 1102 zcmV-U1hM;xP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gnome-stock-text-indent.png b/Media/Themes/Umami/Icon/actions/gnome-stock-text-indent.png new file mode 100644 index 0000000000000000000000000000000000000000..6da223e27f1c96b515f761fbc7cefb4ec9ee7011 GIT binary patch literal 854 zcmV-c1F8IpP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00LM^L_t(Y$F)?=ZW2)# zJ$Lw7AcU$kK}HsKTc5-y&{bn=vGFG{rcm|`IB8>}F>0+3pnDVF$AJLSg{h&GB7Dci z%)K*|5m?}zOmb)L%sJmV=erm9&qL|S(VN3B!{PA$GC-5bA((7!-}LeN_Q#9tu+dhfE?>l1iW1O}W27Nh_GJ{-S~wB*rd$)n`- z+7tmp;Lz22#bc-R1Y}S#AP}vj^SFzX9gBqX}spJ=z+ zXm`B3)9#?t>7aMsGvC!;)lschQM8Njdp5F7BNQ-XkM8&A;W!Q&jRr&rQX)6nbOz>F z5ChJCY$5<~4jeR_&Cq^XAY>pYoU004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00K%$L_t(Y$F)?;Zqq;% zJ$4?8q)`eAi2)}P{T#jk)`-vskPs5cG_Z-Gim0fl(t!8^*dy^jPNXd80u_Zc4apo9 z^RV4GjThlsmd7*pIrrRiXMq14mL42^Jp4MF&F*aiG@s9Zd;aXj{n%vp{X1?d4nK6M zakH1%FQ(IoE`kD}xRoB%!v{^xAxlS{-5LU+95W_Vro2z6eS`BU=e%9OzA-|4A5}vW-jW7>9B-^>}3Mgng(28 zFbv@P9hEsjiip`tjjUga5-I7r?SkO|zTb&gAX1)KtEMTDFh3E&U;zJZ$E-ctYob^z z!f_lpH3!vN6;9PbtyV>)awA=v>dpv;aqvWy>M2awQgo3758(Sgt_)KyhTf$$x#2eO zru}*?dbG==4@^TKKXlzX+OOIx;Z42_4~lgAK=(8Wx9(!!+s8ro007YMokTL#C#&>A zFqvRUf!Y4*2iLs~&-22PL~pGaD-NPW1?JR}8x8>6TQ0np9$?gIt)UknRg147*xtUL zm`SBlMVhcIEB0;(a=G01Ce-`TK8 z02)&bUo+vKmH+?%C3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uR zFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGAl4JIxsNj SSzM$500003k;a4 zTN(zm&==4`c|YYfBsw0Jmka^79~hPyHX8{0{>Koz<(0#*XH}Mbv*m45HLja1;0SK=FyaoW_V}N`j zz>@?3q!!>5ui;i06#$HRfiZai5co!yW-`11jD8LcX8HpFl81qJf#hNSr&m4(K{3<< z43%&2Z%Z%qjV~~?dvV`Sw=@nUDM0?d_KB)7#oPwS1g z&`LDn|BXd4m7}rZkyIoP!>BtP)_5dBj%6y0nJR3g%si?U7p=iXYjLsdCUJ7hxK{JH zPCQFt&FaLnbw}A2Fz6T%QYx-ID@dTY^;-FcQ+Ae9>jMHq-xpg$T*@3J5kUMrP zwbdzg()`M8^E8cfnud@zWuC5e$((*KtDT&qbIa)>@pP2Wf3V5zb49zTXb*x8y5&!O zoUi|&pv#-z{Yg=eZ_$9y7rp)^eE}tto+S(COZ(}i1D8sN{RI6%SBJu{-lJB{ek#^S z-CT(Jb|&e&>FkDSo>ZSD)$?R``SSG)#Z0MkR-l{{D3=SAKV_=s1gg0*)tpduPoUBZ z)Ovw>UZ9>YYhKH5SuAT=Dr+@R*t#s#tOzw5h1&TF?P_`3YI)m(^7f^Q_O|2oxtu~CTHI1z)$JQ0&8>)#7^~6T=146k&=gdV@=pejN;6Si7J1K0%LdQ((`MRS7{r^Iul33(X3@GvWZCSH81acy z!6r9uuT-z>5Rce#>@N0FJ#B_!s&}2n(!`Mx$^b<&JiA+sja$2Z!Q|=F_oZxUuN!Sm zeCEzsN#oYW)3^s74-K_!lUBCTyx zvrMY&IGlo;6e(4zo^kcqNZF%nzovh)s}fAo^VBI<<#p!m07ygKfA7yvk+{?TZ1BFaTLx!gpQ3uGdh zNFaI=h!iHt19828kSOPgM1)9`Jog~I6UfR*PD?2QGLb|g5Ge$bJCj64$nFS<{5J4w x3`+$7R)!4`IVmVVF_#N`KHnuRJ0p*i$m6=?+nku{R{Os%dP+b literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/go-down.png b/Media/Themes/Umami/Icon/actions/go-down.png new file mode 100644 index 0000000000000000000000000000000000000000..43e99e1245cce419954698e0945a2ab263f72bca GIT binary patch literal 1072 zcmV-01kd}4P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 UXPfRk8UO$Q07*qoM6N<$f`x!9Qvd(} literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/go-last.png b/Media/Themes/Umami/Icon/actions/go-last.png new file mode 100644 index 0000000000000000000000000000000000000000..0ff989430954bddcfcfc0d8e1f0cd9f4f7299b3d GIT binary patch literal 1106 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-bold.png b/Media/Themes/Umami/Icon/actions/gtk-bold.png new file mode 100644 index 0000000000000000000000000000000000000000..87bfdb2c7ca50dcd52f718f6bc29454679204007 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00S~fL_t(Y$L*9)Xk0}c z$GA{;!VXIOb-D?q0rJ>X>P3` z(yGvl^wwPb0}_grgl0|Hy4k#auibt7=k3ee-FY+Pq1%!_ZfL#ep*_?(I2BX@yG+F3i1O*liAVw9gNQ2QtL+E{OZFK(SpseYSi#oz9jqN-6Yx-K!3!Kb60<|wBEkN z{Ok#@{UY1Q*7dOi$w;6l+=-0oAgh<~+wU2YGs@7dI_$bcPc#&;b$x92Q)r`rLTBE2 z^-yrdsG-gTmNTXe;JZJwrp-Ko)x3kgL@cOMI`h9cz+Dbxhjff2X<|P z`O5y&j_tB?bpz#E6V#^xk{>*F=1S_R01743Z@>0}rkCodI389rd6#?a;xE@vxtz0$ zYdOESv1lX!$mu5rj7(gP3*ntSKAfa?G8H&=2IGE_Ky6+E0I1DN4~iCX8w|{o z=nh8}Da6T|R!rWv502byZ3q*<=rul#~Pe6d8 zGZd1dNvHk^u*IgMqqC|#&}V-1!5h(>#h_btEG{iOg+ej7ec+yO#OX~wALLxZbvKaD zrmb++NX*aAZn5l^hgR&5jqTgp?FUh@ZdDQVs~CUlm7wNR5b!C`{1lpBLC3=g{_yiU z{}3=i!OEw9(mWD;(p3A5Y)7ZRS3;9P4 zMG{@i$jr0_42_>pDU=L$hV~$P-$kWb#nPSIo{+NbcR~p9dIk=wel3WOP#BgJgG1xz zQ%m1|a(f3rrOI^jxuj;48fbU|xwWjzh5T~y+Q(nC1H5$V!r7{|{#m^LPzONJ73tL~ zmOc&Oy&VAIv6s{7yGlBJw{`o38R=Ws03-k?t$1V#34rGpfBe4p#Wz27N@ho^T_KbX zxBAH)F52d3^iZ0hzmTT8pQsAOHXWC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfd};CU literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-cancel.png b/Media/Themes/Umami/Icon/actions/gtk-cancel.png new file mode 100644 index 0000000000000000000000000000000000000000..969d7e46b03cb6c0658412c95bb0bd6c4dc74e4a GIT binary patch literal 1375 zcmV-l1)%zgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000XU000XU0RWnu7ytkO24YJ`L;wH)0002_L%V+f00d!4L_t(Y$Hi7_Op{j_ zK5haMVv5G7KT!GM0u)2MfdYyVZ>V@zRFpZHR z%jPatq!fy5Z9xQFTW~}u9h94z`c>WK*>gUtS!SuuA6t_rUwXdtKF@o3&S{1Czd!E_ zF@c!q2QNG~d7cnQbA`Cir{W8IcaP7{C^F)C$UpolDaqqLaRQ#QGI&m(hNrX?o{|zM zr%pjRejImlb5Wtydbsz~ebhH#X^Q`2>=i}oQ_sNqgdHeeHXkWMPp-5^f>?>BlzHAw6ixq^8u>@(Vq+DVxntixv~Ab`)3$B6N|N@qYt?tYgFlDx8_THG8iCC@TwjT3u{P}2N_KnQm5*G*4tXUY?v&WCXv172YWa)9+k|iq46se}0y02fy zKz~1OQDBf2)UkC9axODw;HPQRP#+P2D^XDxV2g4C0)`RD&xe(^lm3=MW(w#zhi>*? z4%AF5*3nV7*rwYQ(XragF)^qT0+oc0aB+4#)xHJXd3mt1Y->^ynkb~Yg_-wsbf9kU zUeuECLSP`S(8h|(g}Y8CzU5#qc*(2Ak4JBMx^Dq#(;@cHrq!Y`G!&+g5LC+(TV4)5 zopW*0Bp4Z3LtG3CLp}F8E?q(;MSqZMgn@fphYtA`;Ni&mWZXDqfY!5yhr`58CT65( zo2sTye@t#5^u*V!#F&=ih4jU%W$XptutfCDt!!WWk5Xx>xh4n zU;bOqCApCaIe4 zsW<9TA+pOf8l{*1Z(?6-XgdR;C#pEBNID=lc%thb#(Dyx@(XYH=om;k`n``A7&4bs4(D8ROj&b3_UwiloAtH%nA|p+R zZ{8E)H{$nqh4_Q`^BuNDKIM10{<09w-1|0>Nu+q=1N@1M@Cr>NB8UVcm3W1Cjd+8| zBGSEtGV_;+2}IC;{%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG# hF)}(bH##vgD=;uRFfiv?T%-U1002ovPDHLkV1jirW&{8L literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-clear.png b/Media/Themes/Umami/Icon/actions/gtk-clear.png new file mode 100644 index 0000000000000000000000000000000000000000..fa8cadb5726aacd5c9d48ee4062c1cd72e5d0145 GIT binary patch literal 1381 zcmV-r1)BPaP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm0000;0000;07l7cJ^%m!24YJ`L;(K){{a7>y{D4^00crwL_t(Y$Gwz!Oq*pK z$Dh7!xdg^I#^E?biUC{#LUckzNnpAqGsA!r78lWpOlGDmB)YAvfWzs=WCNFQ6sAsO zW|K@HPNH&+%DHi0r5r1CRA7|5U1`5Q?|kDQW(o`Fn|%Lyp5O2H`Cacr$p7(u9>Z*+ zI&vUbzV6H*(niP zFNC7GC^l1i%Jsr`b20_Ct->20h=sg_2~^A_W{ONFlo~rkluez*I_#sP# zlCut&8x!Mtk?&moaT2u=+Z2_(-(;!yq%F$xTrsVP$D%R`^-93-RG9Hm^R5qD*@85cr=pOkht0l1nkv*^&9DdOoG86_?7z`7U&`+m9J@5hYC5(1e0L>z^p)A-i*0GZU81z>QplsSq z0p6JD^@i?t2o^`8Fxl=8{rHO%um=XsS=5&XYw7T%tG0+{5d8J07V~QPVaV$&&{xGk zal?uZS%j(XKxlgXU{Jq``QccMw`|9~=}dIjMC*&v_yg3}e6{!t)gyT+q@|;3r*ZVw zvwv1mWdUWQ17;OFFmclt^ZkJ^+&zwIc{1v+_}`PJ3c50rh~sL3t*M-;y!P!PF_%8G zU7i?t4~pNMFx=>Y{$H#x+37`}kT)!kh<-!N`{XhTy0AqSkqHJBRMgRZ+C3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgF)J`IIxsNSu|_!n000?uMObuGZ)S9N nVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjfVDxHP literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-copy.png b/Media/Themes/Umami/Icon/actions/gtk-copy.png new file mode 100644 index 0000000000000000000000000000000000000000..94c72fec54fb6c3bb50fd7a13d797816e8ab31a2 GIT binary patch literal 891 zcmV->1BCpEP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 KMNUMnLSTXk_mFk~ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-go-back-ltr.png b/Media/Themes/Umami/Icon/actions/gtk-go-back-ltr.png new file mode 100644 index 0000000000000000000000000000000000000000..90ecc8b397791e29e3ed1c811a076983119527e5 GIT binary patch literal 1135 zcmV-#1d#iQP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-goto-first-ltr.png b/Media/Themes/Umami/Icon/actions/gtk-goto-first-ltr.png new file mode 100644 index 0000000000000000000000000000000000000000..1822571ee05a1649593d5f07a36501ff2040600e GIT binary patch literal 1140 zcmV-)1dIELP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 UXPfRk8UO$Q07*qoM6N<$f`x!9Qvd(} literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-jump-to-rtl.png b/Media/Themes/Umami/Icon/actions/gtk-jump-to-rtl.png new file mode 100644 index 0000000000000000000000000000000000000000..0e677738c6e7cf12c0e4b65af6c9c212d8c475dd GIT binary patch literal 1258 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YNKL_t(Y$L*C%Y#T)w z$N#gt_PTN6P&)~wd9)=`9t{d1jS}Ke0R;pUQmI8kND&7vAR%s{hzlnqE@&@^6RO~X z1Oy2LNC<&QMXRopme9lm;wDXrvki6}KVm!aa~>SJu~v?45E3^&>95(H{l5QiW_D&5 z_&@$8NTp8iq|yk2*JK-cQZO+f1niPbwul6SQ*ntiyrXM0^{)WlNo7Kk$@{&1T?e|m zyH$_dgF3qlnanPdI|-~TujUti_2)R{ZveQUYn1&hdH^&|ytUJ`yiaNOw z8Mcj^yH`-4yHJWQv?x80vMyY_{9`5*F{ZA(OJAx5kS!x0x~ty(QFnibV|g*Sb9s73 znP2c1vgtyhuWihh%El2&`~f4jf@CI!q0UpbP&m}(8O$yjJ|3t!1Mj3Fsj5BbR2y4U z*?5o^*{c9_0F6Y#%Z+vJ*Uvn2PMwc@i+E}ag&c=lZ9`x4!}#Kdk5hv4ibNi^Az_0D$3YbqWBmqR<7$oaKef06qtB)|woR zZi)Wm50vf;EC+{Oh2S{mom51b_m{dx&yi|m@1&v<3Dce?_lZY`pH&@-^Uh2l@O|I{ zxVW?ZpYDC7RKqyM^wqEPn~}{q!Es90Xuc}IggA)=y|uNqXKdgJ`)u$V?Bo;s^6J%t z7&$VI8UMw?x)BTr4zI4!-O?_q7KV=(+ef9(Qt9;bMV569c0MA7Q@22Qv6SOboESMe zj`_eWy|uO)gus~fGg1>^LUQe+(o`%Fn^5d>{e4G9Nhoy_Trk*V6(dK-u@YS3zu#C4 zL*PMOqfu*pEr4V`R}3G|yF1BOiA;FH)!=e;-7`oc=`{?spGGM1C;!ENHBrn6@9cYQ z-(f2U5OZhQOxgFYH~k`8e+b{jbGKi(yXmmKzvCo{#KOWi_tVZ!NIZ7^L$+oPiU1Im z0Fr55t`aD^B+||nT#Dw_pCdACKQGK=lk5D<;-@TN8e1hQ0rr-~D#g;;qc_C1vqj7% zY51w6!C(Asid{GBt>-}OiIf548j`G|^K_BA*s@3iU=>77?fc=7GDEAeRF*FQa7$}7 z=T^gNK`NMA*S8!{^UwC5#b5Vt#G=6%pKkyF03~!qSaf7zbY(hYa%Ew3WdJfTGB7PL zIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQ zH&ih)Ix;spF)}MKFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@W UXPfRk8UO$Q07*qoM6N<$f`x!9Qvd(} literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-justify-center.png b/Media/Themes/Umami/Icon/actions/gtk-justify-center.png new file mode 100644 index 0000000000000000000000000000000000000000..896dfa106be7b47a9277f420976407891a146bfc GIT binary patch literal 694 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s8-o-U3d9>=#%KJ9nNL8R@xx%5qAScq0VFK0X#brQy(dBoX~nOL{IeD4u!q`iBTU^Ci^}ARR8>s zG4qG>jd{zzUy!xxOO|sAxx8ij>(#T}GBy;Kd$)JJt8I^oRrQM36Q60Uu5{vgzf-c8 z+u9ihf(#5tp9wLZKd--1VcJ!uMIsGlzk`&r_ZRxTWZ2zmKF!5(CBr7OnagJKCPeBk zlywmhP*Hevlc8th0X3E*8Y~SQw*QwhzH?G)s{3EM+O{o`<(!h;`i>*bYj5Uo=y>K? zGtMdY571%jJQT!GemKFP#Uya`RfkuqyG1416jvx|R{z(IWlVl?chcEo8yJ+_j_iCR z*|J%>*DZSO$}oMW^-+tMPxAHBojmYy7_4tXcQps&8M^+Am*ciCe^rTP#mjic?_{6`bCY zVSFs-*bRTZ>6=AEXJ*c_(iZFbR$=pRmfwDHg2D?NY%?P UN}v7C2ILF|Pgg&ebxsLQ0H3x3jQ{`u literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-justify-fill.png b/Media/Themes/Umami/Icon/actions/gtk-justify-fill.png new file mode 100644 index 0000000000000000000000000000000000000000..b009ee3b3cb4edd08d8f5440e80ffa7236b4d18d GIT binary patch literal 683 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s6sPZ!4!kKLiF_Uv2{WI0`V#dUp^-QVIFTqX&hVB9^arE{BB;Lez7$}7UOv!6zk zyjee4rg2#&V}iMWz+n}JGaT!xBkI^!35jyGeq}ksvo$I-Xwn18i7hTJCiR-TkZfhK&LeuZ8|^XA@-R-BG?$)}Ul}tZHY` zPUo4N99>5zFP&6T#A5ybG0W;$A-1*?pS_w&tk$aZ*PCx-X-nK=YqPEIU+X@F%K6XZ zCY_agG4t=4xc`gudEJ#?A1_?L+w$X@$=PeX zwHdRH8BF!Lxi{{8vB3{1=R5j&mp^ig?Yf=7@?Gd20|SFRdP{kVo554 zk%5tcu7Rblfmw*5g_W_Xm7$rofq|8Q!L-j7hA0|x^HVa@DjAFn3=MS+%ybRSLkx|q zjLfYJjUXDHN5(8+U|^60*$|wcR#Ki=l*&+EUaps!mtCBkSdglhUz9%kosAR&0|SGn LtDnm{r-UW|CRhRb literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-justify-left.png b/Media/Themes/Umami/Icon/actions/gtk-justify-left.png new file mode 100644 index 0000000000000000000000000000000000000000..224d70023345320156b7f57b56361d1cd02092be GIT binary patch literal 684 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s7XPZ!4!kKxye%Yt8=p+xlL(soX1_(z6=1HgdUd=gd@`Y8CYD zjPBN?LuVKmWU9A|T*%t1!W6Q4D#Ieq6<61p#GcnvVoJNf{(|$oQY1?z^CXuiyPjUy zYqlV0XN+WfiTr(U1_K^v!Fi%gNg_-RB?mgZmIf*8s?d{WIJ2fF)7Vq%pcm`u|i_Tee<#d z7haZZc{Jm`qx=D%0G+rsoh?dD(Ip$|ydE`mGTh7&E7NodndjV)8Y%C8{_N+uSDsHe zU!W`g^Uutg)@SE_*;jQg!J_0}&Hb4^YUfJ4G8bje>gb=Td6-8`sZa2tuv3LoNBm&} zo6AP*+j4o0UiMsnTekoA-}!<%vv1d4w@(q{O@E}6@REUnLAAs+q9i4;B-JXpC^fMp zmBGls$UxV?QrEyN#L&XZ*wo6| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s8do-U3d9>=$ap7uNBAhNeSqkE;mVZP~~ z%(GrF2_?0j(3cRlKiJ}-BDJ5Pxly&VbN&%0F}C2!kR8`qHA|v2WY?HyTFp&6xAXaz zGa?V>bMJM0z42yg>~|&3fWrn==GV7xGr75A|KlaAeqKo659i)=KJD^e>-02_<@`%5 zbN$x{7#K4!wBK_T+Hzap%jN2;rBaT2>$Wb={?F>LWQBrMMW*KjkC_ffmw7E+*toX(+sd#21r{&OkMm7C zG;Z$LbNmRmaA~^EPU?K6r;CYTW8|ddp{YO zVz1wyH!nK9-YI_ap7qD>EeNo<`04o1#Sf2PknvZ0*8h6e=bv|CbTpi|oDle^Xq-CJ z(fx>|l9dyOJIBYlUQ1e*kj)2?C{7#J8- zOI#yLQW8s2t&)pU6H8JVj0}tnbPX(Z4a`CeEv$@9tqjey4GgRd45odyFhtRio1c=I zR>@#wU}&goV5Vzm9%5)@Wn^w;Xav#lJThhp0|SF3$cEtjw370~qEv?R@^Zb*yzJuS c#DY}4{G#;P?`)(P7#J8lUHx3vIVCg!0Oe{Bs{jB1 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-forward-ltr.png b/Media/Themes/Umami/Icon/actions/gtk-media-forward-ltr.png new file mode 100644 index 0000000000000000000000000000000000000000..6212e20ed5c18931b307941f1438e05df798a6f8 GIT binary patch literal 1229 zcmV;;1Ty=HP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00XK?L_t(Y$L*8dPg`{u z$G^YdIj22+VTHmd(pSu)61Vm3fK)>?v2A9Qwlbrc+q_v~vI~NVE?acb5Uvae#p#V% z+%9~{BF+oL)GcIA{{)5!Dlb7OEoCJwU3>a+&d&=AW+1^`nV3B{zvtrnJfH9LOTNH^ z`rj8I~?xL*&&{FG5_Y1JT{K`vXEiIPj`T2!~+5&Z(w(026*y{%#InbNS z=4SKx{N0+Ne+6hf>^bz2Px5tJEc+vRqhTqrlvo2G17MpMJQ3&|^msiXQ769kP^+af zH#@&j(3gm&;}1@w$(N zf?*3U@aXP-TIlTPG%qbJzF@UmeF{PRjPFPx-6ZIyO6bgndRkZ@SSb)K3Bt6eZg|;@AE4h+5Zg<-1E9L4O$8mD8 zSS)V=y1Ked=EjEpXf%dOr3{Yapr{HIr3$axCj^5bLpr_sOfH)>heBZu2vDt7Q7RRY zzng=@?xeA3ub|iK4sYbMC#`nts?+XToSmI5Pymo5NdW)?0+p#y6a|V>LAhK)K`y}D zV8*M*j_W~)o)8K`C@7^6G#UsR4N7Gh9OV!W9uXZ5m(gG_JSRyK1ppoZ{C>YOdGp3! zf5!3PC#W*7vdF4 z>4*J&Z^Sz~I@SOfK?nc_ov1@5lVKzA5qTw<_|fC{zSDoY{})2Yj)e1D9Rrf7B#XyK z#nsnry&e)DzV)=#&Vtv35xmo8qO1W+J^?8v}90J$WW z&z?K;yVKS7=BF1g{rpcus>-s1g9ABJy?Nlm*vFSmCR3IWvTKU}HqY4p@$vD;*4Nh? z7-QTnntfAKQ}&682`6Jrv**%n!#rb5*h51ZV_L=-xv%99>OZeP0p5CR)!r$^&Hw-a zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuG rZ)S9NVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjfT^Sc< literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-forward-rtl.png b/Media/Themes/Umami/Icon/actions/gtk-media-forward-rtl.png new file mode 100644 index 0000000000000000000000000000000000000000..134fc02c9598f273f7f29b75789065e12647ea05 GIT binary patch literal 1184 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 YiFA8rQcU@*^F{ywPj_F+4c7?XKM-$2EdT%j literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-next-rtl.png b/Media/Themes/Umami/Icon/actions/gtk-media-next-rtl.png new file mode 100644 index 0000000000000000000000000000000000000000..070ce76f4b8a0ec02ff44a520bcfffe63f4eb3d8 GIT binary patch literal 1246 zcmV<41R?v0P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00X;8L_t(Y$L&*1Ok7nI z{_dOi?)#ZxW|)zIN<_w?I)Gg?GXb?Jg|--qt);2s6qFz9UkK1itU{y)B3dOH7lwqA zrpARU6Jx8i$_h;kv5UGWi5sISCWN%1fmayld+=tCibIx~w z`}n`%u$>tI;olPQDvDC-bZSM6v3$$2EC9yeym1032A~00z6-Ab*am+US3}5NhX&VW6Yi-BjELV2>>LC0{|W@E>>#x>`s(E5`6Ad|EZKO z-|y3JEg+pv!{y8a0PNjg`+Zw$+j>Joqh^|hdtqUI)4*g|hH07^fr`q?6k`mMD1kA? z9^X-Mp{J+Eda`<#`?j&lmX>q~f&f9tm@X2Dpu6jYM-&~dm6c@>5d?<=>2!KSAR3LP zrlzJczAq1lQ}zlZF1d<|wh6guX4*tW5%uNFJEUg+dkph}j|6*VoT|`2KLmjq7v$?H#SlSAMzjyQ*ku&6eQWA2Y8Xf2IAm z^B2CJGEIZx%`t0w`f3RgiA3a`oSZCcf9b`~ySh6U z7WKtBBBEIQC=pRW*L7ba(fw+?Id<)bA1D8`ENjiq-(gdG4*{eYV>%HTb#-;U0|NtJ zb#}h?NyCB0B!KN%_bm|77c(<67l%Lk;P|=E&Lt9wL=r%7lTCd-za@&I-qz9T=bY<4 zpWoWpHpUDhGKYqSewv+~-T&^;V3#O~dvcqc<*$#8p8a%UV&coOu`@lQC{@|F6-1-~ zaAjixkYtRlVhe~!0^rVmEX5eRxrKKC$O9k)NHfOf?Ia^2fruFH0U}a~i2nmG5s`SW zJooY6;4jwVU^;i$+rt0=03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`If zHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)}MK zFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@WXPfRk8UO$Q07*qo IM6N<$f~q?q`Tzg` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-pause.png b/Media/Themes/Umami/Icon/actions/gtk-media-pause.png new file mode 100644 index 0000000000000000000000000000000000000000..378ab5edb36c8fb6d2bcd0334c988c8d96239c6c GIT binary patch literal 798 zcmV+(1L6FMP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JOML_t(Y$L&UPt#SW>YS>YZs5lLd!wy$A3$7#{Q=+{z!*TQ2J;=j*RsPYAA}IW;nDq5DdnBA zUrPDs^!qW$GN!4U-`>5Q-DBo@0MuS^L z{xt%zMZk5-YHH2yCV>?F_MG+Uuq;w#pw4L}tzyJ`(vF%{FTH^QOCuqY^j_o99x7!wQ zd2s<_mcWLfAWxFSs&u48L%BA;9IGR+03isDqpZ3pjPkMwBA}rfP)gYmb-UgD1pR)W zi`^;JfO1sP0NU-fy1`&z$EZ{Tj?=IrdcEHM5eerWTo!sekc!Md8NWm#C zBoRiV(f+?^JRVy>93>TCx7*DDU^1Cll}XY@UH(pj>2#V4A?8n>J&MYHrBr_V;NTJf z(lq5f&wo9B`mpjA#c{Gq)AZViDmR6hTUnNQ#+ZuVbzSy6&k)gqnOnvfFUztD#`8RO zU3W!9OWfE$Wq$!yIhrA0r1(An001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHY< zIx{soF*YkOFgh?Wmh(6<0000bbVXQnWMOn=I&E)cX=Zr004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00Ux4L_t(Y$JNx^YaCS^ z2k`Ij%$c3NZ?>h$?!>0aY<8Piuh_ySDL$pF+tecDLTcMTpvH$HtuMs~LGTu^Ao^l8 z)h2>}MB)t*vHI2&Vq4OUW=$_VEf*E&wUkDBNK_;iQaOlay6IBRRQP# z>_-AT@zkCZpPrjh0s&tlm(LyT+O?CeFK058i@=k6PE1Z5;hGj>=@-(jLaDGf z6bwwZZ4a$xmREB1mWeF^W258z*6kbMoMUKUKOY<#@Ml(54s`G8dR}5!T3fqm0Wj6jWD9p}2x zAu$FfF)+qJ1em6QYPE{~zQ?7J^g*RiEG9iZWwd=q$2HD*K~+^pRaGOK&29`NlgT!h zBELE|I?jb5z?cNaSR;uDRogM>G<}$jBkLX&HmTpf{pvTBA+;ULY2W z**9+0K97Nc7y}~$5dl<}9S5@Pg2(N}YIYT8rq5Jzchb;jBNfKl(0|4e1=8Z2dd|vVS{BOQ?^7O3N>&?gG@eK!K6VTn=-2iOc1`$D) zCAef4mVR5Zr>EZ&k|bR?_VV#}QmND$5tY#*ngB&n8f(TFd|p3t`8>{^{h*S~{_*p( z&piG5)YQq#0E$Gk>G0#NHWil#vvaeRg|8QGMmsuAo%>{F9zcPJ%*_gK0yy~Q+XZXp z{CVB)_r3l8yVIYwx3}kssPdqNTLOyWzBKpc+~VZKk+;(6^q&vfj*Wtv-raF_pmqUG z0C$MUdYHoZ4aad@j^ij>uk;_q-%^7?TS@`r?f?J)C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML zHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* XE^l&Yo9;Xs00000NkvXXu0mjfMiBdA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-previous-ltr.png b/Media/Themes/Umami/Icon/actions/gtk-media-previous-ltr.png new file mode 100644 index 0000000000000000000000000000000000000000..070ce76f4b8a0ec02ff44a520bcfffe63f4eb3d8 GIT binary patch literal 1246 zcmV<41R?v0P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00X;8L_t(Y$L&*1Ok7nI z{_dOi?)#ZxW|)zIN<_w?I)Gg?GXb?Jg|--qt);2s6qFz9UkK1itU{y)B3dOH7lwqA zrpARU6Jx8i$_h;kv5UGWi5sISCWN%1fmayld+=tCibIx~w z`}n`%u$>tI;olPQDvDC-bZSM6v3$$2EC9yeym1032A~00z6-Ab*am+US3}5NhX&VW6Yi-BjELV2>>LC0{|W@E>>#x>`s(E5`6Ad|EZKO z-|y3JEg+pv!{y8a0PNjg`+Zw$+j>Joqh^|hdtqUI)4*g|hH07^fr`q?6k`mMD1kA? z9^X-Mp{J+Eda`<#`?j&lmX>q~f&f9tm@X2Dpu6jYM-&~dm6c@>5d?<=>2!KSAR3LP zrlzJczAq1lQ}zlZF1d<|wh6guX4*tW5%uNFJEUg+dkph}j|6*VoT|`2KLmjq7v$?H#SlSAMzjyQ*ku&6eQWA2Y8Xf2IAm z^B2CJGEIZx%`t0w`f3RgiA3a`oSZCcf9b`~ySh6U z7WKtBBBEIQC=pRW*L7ba(fw+?Id<)bA1D8`ENjiq-(gdG4*{eYV>%HTb#-;U0|NtJ zb#}h?NyCB0B!KN%_bm|77c(<67l%Lk;P|=E&Lt9wL=r%7lTCd-za@&I-qz9T=bY<4 zpWoWpHpUDhGKYqSewv+~-T&^;V3#O~dvcqc<*$#8p8a%UV&coOu`@lQC{@|F6-1-~ zaAjixkYtRlVhe~!0^rVmEX5eRxrKKC$O9k)NHfOf?Ia^2fruFH0U}a~i2nmG5s`SW zJooY6;4jwVU^;i$+rt0=03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`If zHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)}MK zFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@WXPfRk8UO$Q07*qo IM6N<$f~q?q`Tzg` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-previous-rtl.png b/Media/Themes/Umami/Icon/actions/gtk-media-previous-rtl.png new file mode 100644 index 0000000000000000000000000000000000000000..56e943d7d5cfd75cff894c5d7f2090f1fe7ea78c GIT binary patch literal 1318 zcmY+7dpr{e9Ke4=w!Ar?i*#|ushGA|nng7x^YJLJ@~W`LX_&M&L}qDBrSnXv594ml z=t^F-JW`ELlDQr#EV-0d;XV#GcAxJ4y3hCX{qy_z3J&;s=xS}$0svi4D#afFOumZJ z1ONrcWeET@V#q#ZfLoWem67WJz=B5g_W?-#0f3tYumS+wj{v8z0Fw~_q*8zli4}MK zoB=>H%FBZSs;^YGi3$N=eIhmFBmhwVb>CygZU7B+PYOAZ+C38(=Dmq-;xZb}9=dv% z&A0CCdTL*K#Q(C9v8yCL6jRjrBI%RcdpC)UP%NjI*J(=0jXt6E2Wy@g?A=B<(g2wI}NFJwB_$Tgd9PESwApD|xFx?b9DlKv^BY&>n@>StAZq59iymG0Wn zSr2F}*3g0;5W<+PvPLZta_T!p?;3~Vj(8Sm={)%98v(_5ZxIr*(b5`r_hO4y_;R!O z@zfx&U|%ZA-S79Z*ONPqWAuJudiM`MQisoMY5Mb~OBvGWzR3rSbcWflt7$WME+FCQ z+iyO#mPfI&AuW2u9U=)}|u%&>Hd-6W|EZ7)K0?%GrkIKIu5 z$u!rO&CB0Bp7zQwrb;?6s90cKFt6js?opp)PE~ zb}~a%(RQoq9o3Dtx-6H=FM<{VbQG<0Ore_4!YV~p>A`w3e5bFte*3~T^VWzr9n8@+ zc>@w@YN|%lO3$2E-lAUEybHVdtzBaw6mb2R`3ift_GuE>jqhC0z)G_SKf0)M3{_H~!IWt0$|FkCR= zbgJxjuyxKo)NLp2yEl-bM;FWrUy9moS$9?@Ht@^^Jd|bV8c(TAJG&(%<@S&KV@+o# z&OK8okQtlfNM8o~${n(ad0|$T!5E{420Oukq`G4dM{#9WDk1gb!@0uk$!*ubkgRw_ zw>2u*efgqTT%(?RBqwcU$i@MUzD(fc|NZLAIj=)}KL70=<&q0Tu-S|JQC02P%=CfD zjMc{eiE^YtbFf_kC|buWYil=}A|M!LpL3{V^J=xIrF)F^q5VyI^Muf@cR{?Ah*U$I zj5*XjzO=LyKD06~3JVQAgD|iXqG)!5-Pl+7axGBI8E5nE_h*>36P17TZf}zZ`1v_$ z??qc`0*yxd9#iMztt8EzILcIUtvtz4MbPg|yyHy9D0R!q^t+;(d_~n+{%ieS=~H5H z$%iNYyEP*+bJo61zJN0WMs42DlYbJnVr36AsGeBalM6WQS%l04Cu=?Ze7HN~dsTay{{<5v;WYqY`t=aNjAn5nljy+VaO`6f7$<3w YiFA8rQcU@*^F{ywPj_F+4c7?XKM-$2EdT%j literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-record.png b/Media/Themes/Umami/Icon/actions/gtk-media-record.png new file mode 100644 index 0000000000000000000000000000000000000000..d13d3ef8b8002b4faa73d1b26db642568e6aefe2 GIT binary patch literal 1258 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YNKL_t(Y$L*C*Xd6cq z#=kc!X=SZewpVr}Ir2Znq$w?N6(u+n(-dNygJ^pQJ>*!DQ%Kt!n$T0wA%@<1XbJ7L zp{3AEO9(l{g%Sv{n^47R>_do+k!nkdEIIXRrPZ!>rUzTZ|CG8t=YtP+hT-?U$2?|% zSNa#x9?nPr0sy=Kyec5;KplX&;WYrx{|)e~K+m_phY_L^K44f9qT3;QRs&}K65_Y> zz)x#H9zf+k1t^LV`V=_)UbXslcOuc>l}yT>zCO6Ry3i<`QgCDi{b7wXIz|f*yS?J#YH&PDwy97ZEVcde&|qF z^xnPG2N?U~Ni@2=y1H6_X+R>8a32O@r*pX<4_>*_>FMt$4>Bg-fcc1~>C|1Ox&BuVUNp2gk=<)rSv(YIRq*sMi6@ z0%|qrlamMz54-)A^+7ls_O=X2rBbeUGnt6HypEcD;JF;Y7zAS=#y~{t76Obh z2q7QQxptij1 zSxF{m0a&f>fYa#c-$ICjT_`|~#Xv;Bwvf$m5J8T`U>6F2BrPi=BYG;8YCWc!rd6C+ zY+~u=O;brEz!YU`7aMM_B}GBokt0~Tb<4B{2fxrXZOfCf%ipS$QA`-2p2t*`=5XHI6 zmp^+pG4XvZpC9Z=CcV5`1uvB#DittQ1q%j2J|A*7ZkT!|v!GnPdg--zd}T9y*AAM> zX0vTwlGOd=`0)#(Q21CK8WKt*B6~EAS>-aX=5qD5r%#DOp)U_kO?|WCIK|%H-rA0W z@e&CkBmkYoYuBQs>(}41GMP9kl{c8GK6dr?rnR$Y)BDbyTLw@jBEB1AFF%}SkPN`1 z>$*!-)yBps)``g33*(hq>L1=SsHBN#{u}@R03~!qSaf7zbY(hYa%Ew3WdJfTGB7PL zIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQ zH&ih)Ix;spF)}MKFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@W UXPfRk8UO$Q07*qoM6N<$f&eigfB*mh literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-rewind-ltr.png b/Media/Themes/Umami/Icon/actions/gtk-media-rewind-ltr.png new file mode 100644 index 0000000000000000000000000000000000000000..134fc02c9598f273f7f29b75789065e12647ea05 GIT binary patch literal 1184 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-media-stop.png b/Media/Themes/Umami/Icon/actions/gtk-media-stop.png new file mode 100644 index 0000000000000000000000000000000000000000..b95ff2480f434e816ee681d819be331ee1710697 GIT binary patch literal 681 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s79PZ!4!kK=EH?|Lx>3j9kuXZbp9*P`xv zm5CR+OT=Bu1uwpK{lX=>*Jq|nMY_j=w?5*lS3g_EuDGi(udJ->+i&aL2iE=DopCyS z|I}YiCX6B56eJj5==0^=3zKN8PJh1q`{Gp5_i$uwjhgl}`@dwT$E@dG3tm3hP|U)Ru{G+}>#t_QCQ`g76*?TA zl;10qIelL8x$d>srFVI@-}@QG9w_qic>;^l0SWGaO?wvX`e1u*w@u=Hp*{Tf7?gU{ zlWSDt14LK6`qH&VeCG0?wPAmiUX|@;Vqjr1K5TG!L#wISHH(PshijTQGctuo2t9kM z(D%4xV#5Jej~>RfRSUWOF4ZY2eN^H&9FWe}w4g!7)5u|FW4GV?=QHlL+jMg<%n1Iv zXv(V4My>^Onl_&aepg~u$LF;)Xqk$Z`-inj8(%zM5TmCpUAJ@BKAUf>>5t+z-;61o z@Vk4}E4zk6Zdb0(61aF=?ETwU{I8{p{x4FvmJ_GJz`&qd;u=wsl30>zm0XmXSdz+M zWME{VYhbBsU>0I%VP$M;WoV{tU|?lnFzvI2A&Q3F{FKbJN(LhXLqlByGhIXT5JMv? zBXcW5BZ!9Qkuggc7#Ji$HU#IVm6RtIr81P4m+NKbWfvzW7NqLs7p2dBXCuYHz`)?? L>gTe~DWM4fwTTSX literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-new.png b/Media/Themes/Umami/Icon/actions/gtk-new.png new file mode 100644 index 0000000000000000000000000000000000000000..f01576b5f17ce433947043de8819af206f7e4c21 GIT binary patch literal 851 zcmV-Z1FZasP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0Q>;y08cd6u?+wK010qNS#tmY4c7nw4c7reD4Tcy00LD>L_t(Y$F-EbYZOrw z#eetCY$OSOkWDf!f|Z4cl^`a9|A0*jD?72Z7C}rSDt`PE7B*rb+lwLy>V`B1wKChp zC2=8$tn2PhGS_0>d~7mnva4PgX67>QoOjN7ci=yl{0oq+jRgTpKc@5W-JH?U#gB#+YF9+vh8W z_iAfn;i(@~yMEEadcBV4d6*d@f=Dbh{|rV+8cgU5ywW%}b_wyOI5hnli=JH;`@Rf( z)WTpKPy%LVV1}7tkzod~jYs z%K!n%XCR=0Fowx7LdtA&it))3wUr;7IobFC{2HiWX}Q)PYEDghk*M-8oy5H3$k=IG z?Gk=aZ71_kNO11#9FwKDis2LlL9nND`>j5LD2nbpcy#a9-vEAHYh3Mf z;7{(1$YCR{8?Al=KB$&GFYmhv0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>% zI65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(b dH##vfD=;uRFfi7!MmYcg002ovPDHLkV1iNAT0j5* literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-open.png b/Media/Themes/Umami/Icon/actions/gtk-open.png new file mode 100644 index 0000000000000000000000000000000000000000..9e4a9780c66fb64eb859c69c33b27db420c41580 GIT binary patch literal 1093 zcmV-L1iJf)P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00SaPL_t(Y$F-JCXk1kk z$A9;|H#13Wqo7&XfD|PK5nQ+^twpSY0Y%){DQ;5TxT|ZI7FQy!-1xC6qO+Q{U=)f% zsc^7zf z9Kb}xZES2HBB&~=ikab@qgX7m_{<_^){6q5R;wW*JbwC&`{vuP&yNLo?@d*mi1fnW zYXDIcvHbaGc<+18Pn~_5EX#V?RMlW?4n%Sj$?@Kk<@*2;0T+S`SzLUs@9hshqo{;g zpl4sXJoV-AwTw%bv=5MFIe9LuZVS~PgdhmIMOMJ>et{R?yt1G4fja1N(3xjv!8K3* zd7^e5I|k1Ba^SsZ>$dX5W5;>^%!!_3r|ubR2dCmp!Z+T#G6&9vav&nasWCTQVyltx z$4;9`Y^o4)-_Zg=7&OOK5v?2^ysHETQ4mZ7P=hq{1TJU{0rI?)Ahvgn6o6yE+~BrNXbS8CzXJ*E-%}005UK#FfA}SEig7zF*rIi zH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-Lg zF)J`IIxsNSu|_!n000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs00000 LNkvXXu0mjf@Ui9R literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-paste.png b/Media/Themes/Umami/Icon/actions/gtk-paste.png new file mode 100644 index 0000000000000000000000000000000000000000..8a955039f03467324658ebd4042a8c4900a3b395 GIT binary patch literal 945 zcmV;i15W&jP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Of~L_t(Y$F-GDNK{c2 z#(($CpR%-3#~C#d0zt`Lm& z$dUj<9?LC-d@3cC0KaRxwEEUA(cD~Ll-=@jnALAH>iGJMRDdQhmDKJ$d$;>K9b?U9 zzVT5$>A%q82=2J~Y_zj&?BaB~XlZH9@JWhDuCW-buC|Garn8pLQwul2Fv=Dp3cDJph@W zDS;USXRQ;f#lq<5D2YU(D1g;!MF@ezQB5>1iryG2WlRWx&1Nfku9V7oIUqo8oZs)? zr#STVc=sh(RzYuXZ&B9CM-A=QW78;^TS7rLOiD>lk2jC8l*zuQ_rjs@;aikjD7Qr`D?c&KEuzsV zN*!j%MFC_=fnsUt2Yr3lGjX~eIV`|#UhzzeQml17f?*h#=H|aB$PRPMGzo=5sMOsz z41-{Bm5G-VTxpmq50D=3NxLMx7N)VWDRb6Tkf`gCyy8y#*1;7t%P)*Q93~pqC}%gC zK5@blFW;h90b)a!SvR*6uJfmLI@O1~Kt&l75HS;g3KeW;?e-xd0mQ9ruV(EY;ao7yg!G&p#(0;mRRfjYnixKn-=5Ct}XH6RT90@hROrq4IE|HpBZ z_6HEIhl|*D1{(kX03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eN zIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)=GJFgh?W T*0Dx800000NkvXXu0mjfXQYbf literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-print-preview.png b/Media/Themes/Umami/Icon/actions/gtk-print-preview.png new file mode 100644 index 0000000000000000000000000000000000000000..760d4c62798d9e72616d6b121c0b7289b059d187 GIT binary patch literal 1291 zcmV+m1@!ufP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00a$5L_t(Y$JLcvY?MV5 z$A9y++jhI#i>4R4BwVCzjH0QOLIRe?@Sv?cXhGCc#h941S*b+~QI?jpm`F|Z!50h> zE7TC9(2LlLAvLy@fTmmoxfEzsXzh(lPv_>5kdTNF0-Mc7a&mHLHl~5JnZ$QgY|lD% zx}D-;JMCSgd~~#rv)w*QOG;_{_An8V5!h@tG)?y1a!&uK`+taf^&VaG&76Zy00ds zsI1(Pc}IYhl$5J|ef`W&il(c_&B=~_nt$o$(Rmi4@DddvaJz@dTDoj%4cg1f9feFe z`jd8)l{*ThRH>&NKZ=x zs3|LVYzWm23l#@yDt7FouKo}vAz7ThkmFq=oWIn`khHS)k#LT;9wAH%z{i8pk~Gt4Y^%3G&Eoe4`=IJ zZ*$?&uN*#DgRirh>OCKlm%E0>rf*o1@fZmSiLA?CPu2Uo7#QdW6#MsAGdS2ksiu=U z`FeK`Jv}{GtybFGPqN8g!p)mb+K#odp`eh5A6ksx@29b;k&Q3Eg2iG5AT}ljzdwM> z<-+gRX9ox&P)f10vXX%A$1o&PN({rG1oVKPQpFpMgvsv)G4J1=mBndJb1mMXkh{&1VW5wS<^HkBTUo!U=l#rbrcG# z)k0)cBx7S^pumuFJaWSzKK_1`QYdL4l^kCyg_IJd1U{cHw7xPF@Nx6^M~j|%GLIwQ zHE}QYz`|5IyRP8#`I@Ey{DHuy5hhd7@UWYlRZk#%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vfD=;uRFfi7!MmYcg002ovPDHLkV1i-P BOG5wv literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-print.png b/Media/Themes/Umami/Icon/actions/gtk-print.png new file mode 100644 index 0000000000000000000000000000000000000000..0c35a115626b7974612b3ef4eb2d47b3d15c8f1b GIT binary patch literal 1030 zcmV+h1o``kP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Rg~L_t(Y$JLe3Yg|S)8F6E;^IAWxg6PSmc_+;IF5tXdRMGH5!|@(EkO|A`+nk6 zQRGMNkq4jfFt z(-Ni6>_|{57e4~-r!#OI2gh+ZapGl$hlkNxC!VtV@!9UW#T%|qeEJyxjYb0iYc0kY z1_z(#`0+t}-^cSjHa0erJtKtz!1YqOcrlHuwgX>Zt1&x2&)n=R)|wsf=$PYBC=__@ z^;0~2_!qanzXkB|rHRj{9+yTQndf=(q~^M=xUMV9%ga(K7n6SIsG_xcjrqGjbMoY? z96EHErKKfCM@JEnL~mnEl9gCN=}pNeC%<8A?3IqdI|%+pT$-35j9M6DH$ep?o&+K`Ms8=(lkLG6 zOD>lqigqBV*XxXo6gV=_PkpV91|+Tn5h091{%JI5Ha+w{g;ENo+R|yQ>FvpEDS{4y zD2fn-e11Q@*pO^uAl6`|ouP@5{uayQ8g3<>D1!{Hemm`hdg0QQ$e?+1PEo#{vz& z1^xi4Koc#&X1Fth*w001R)MObuX zVRU6WV{&C-bY%cCFfuSLFgYzSHdHYDHjtgI}7Kp+$fMIw<{ zEG{T0kVqslnM^L1E0s!>N~Nw))zs89Ha2QBnx>{EtyZhk>2z({);3*xd%Iq*@9600 z?Ck97>gw(7?eFiWC~9C}z+f;84h|ZP#-X7hlgTtZJUlWoGCDdsHa0dsK0YxqVK$p5 zCnu+-rlzN-XJ%$*XJ_Z;=H}<;Ef&kd!h+RmU0htW*=$QoOLn{6;czT3FR!eutgfzZ zZf+iURG$pMOPZE)5l*hOz21idgTo`(oHKFp*`=k*RQ30E08ZKG5@NZvqLN?$V@a`` zbnioMAGj&|`Cjgn?e|++sBo@Uob4yMyV2Ggl*w~`xhKr8s&=S&d#cC8Y4EOyW_^L~ z4ElO~{1N|33*~1$*~3@3mxuPf_SbN#GZk;y${#EFJ3!aI(aJ}P>^5h*Phnlan%-z} z^ZHtS?axM}!`D~f=6ST{`}{i5Od3u zE5e72&>nmD3$}~T%|!99K#CpG^SdOB?b49;_-x*8is#Pubp6r>g)BhR5`0m7_4Bx` zxY7wwY<~I}-BI~q-$ww)q;Z#|xqN9JE9ZJ1Uz!Id!6X<&6oUv)CnH&5r&wh834&k| xM4c^?{3p0pm@CS^1tvj~3?iICMx>KW7BhlHlH0+6K0az|QBqt=Y-q*$ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-redo-ltr.png b/Media/Themes/Umami/Icon/actions/gtk-redo-ltr.png new file mode 100644 index 0000000000000000000000000000000000000000..ab73631109595c31b33c4d71ca04bb3fe46740e7 GIT binary patch literal 960 zcmV;x13&zUP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 Mp+Vs!SpZe=FLHDF)c^nh literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-save.png b/Media/Themes/Umami/Icon/actions/gtk-save.png new file mode 100644 index 0000000000000000000000000000000000000000..747686b650a599fcb879f9faa0468de8574a7100 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-strikethrough.png b/Media/Themes/Umami/Icon/actions/gtk-strikethrough.png new file mode 100644 index 0000000000000000000000000000000000000000..9b0c115e905091c0c354644c02469c261c1acf7e GIT binary patch literal 948 zcmV;l155mgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 bF)}MKFgh?W=UH5&00000NkvXXu0mjfY-sKY literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/gtk-unindent-ltr.png b/Media/Themes/Umami/Icon/actions/gtk-unindent-ltr.png new file mode 100644 index 0000000000000000000000000000000000000000..139ad7036d70a961e0e38effa660777b5e384ef3 GIT binary patch literal 840 zcmV-O1GoH%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/list-remove.png b/Media/Themes/Umami/Icon/actions/list-remove.png new file mode 100644 index 0000000000000000000000000000000000000000..51323d7f9ce43ce709794be4bee2429a564e0e29 GIT binary patch literal 394 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJ@LmUDMkkHg6$}gvJ)SO(As)xyUf9Tc$bqNz;eQn=vju6b zDJ;hpEKOxO!k{B?(TH7V0*|9__k@6?{URkV*6lEwWPh^vYsof#Cyvw8i_U&4c-^?{ z)7pjm-`)QIY?VNs-f`K*-7G!Jay)*8GWp%L-5rq0@P7Trn%W5mVs{*fnQ{0@f|{E6 zyA5~Obf-^c<9Yrr@UQFCp8|)H&RE15-hNp004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv04M-y0O^(@b;$q#010qNS#tmY4c7nw4c7reD4Tcy00WarL_t(Y$JLcfXk1ko zfWJHU&LhvJ22%}~g0+fbVp347n6?oVL~t-To_}E{&CSiL^y>h*+$H+@y0KQURuHjRYnS)dT0~@p z&H^wxn&sX1dI1<8AHOvKpjZr+jfhBfU6H98mmMv2Y->v5c@B$qf`pG`0z!t*$Gmq z2=}(u(z$0B&m8FC&+G+Gjh^A^)hifdXlmNd-lv}A;43e4?AVVOt+xW$>f5snYS~fC zzNfpH`g@8KKaX&3>em%czE4C3pDN7Vtw6-iP_~dfe zsIT{EsBdIsPg3Oc&_Ug9&0;?LnY_6fk=Z|J5@zS8Z3PJ z_QxDv=Rtc%JK2fsDqY8mXF2d%4mV~%iij;>On@~$M*C>(qjeEgE~26W#o`P<4G;Ij z27vdE_9F69rN2;_-0V)cJqL8TUQ;q_Xbx)v8C4p7~ zKF?$_hc=4IY7hlP2pSO$*wwSz{j^yV{Xl48C)AU-$kx{-LFE zH~=Yuy$)sxtFB|p1yoohDhHIy0ZPY-*Z5Rh)W%r^#DV0@%(dSNg{6Z@3rGlN3e*w zSJa`T6KySAZL62n=FFCan|2cPlw(G^_CkY3yhid{zdN^ow&(8o^ZE0sOiGNUIC(e$ z03|+-#RdTPeaZF!Fw?1h3;?8D<_RW1_YG%)`vw5`cZ> zIRK0%fWw9Ifm0saGi z_{~jqsc(f8q~mxcOb(GLme;Yrkg3jLUl2D#{$e55W$>Jl_uibgBZqwDB#HPSVrDJ7cj zLbKgywinIyp@l)TFod3upy$6KgsK81$pG+{n4Opm2OOM2Si&o1SEY@&pDi0#xNhk` z1AtxjXR)knMBn%m0K4PyEN05F@f?mKP+!M%kW(FR3B*43wKZuss+=h%S=B*sTaCQ<0?X1VwzqG z475XO^@kc~^4!C&M;|SZZc{YXd3d_Iw#pk)+b7>6)#|CF16=96V%cJ`)EpI2n4f410-r1910ADdLD>62Sa=E^$p{H!;NizHjDcYdo00hcg0mu? pP;e3G7>);F;X!yr3QlLxBN#aS-(V>t>)!rR@iB?4?>^yN{Rb2{*FyjR literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/mail-mark-junk.png b/Media/Themes/Umami/Icon/actions/mail-mark-junk.png new file mode 100644 index 0000000000000000000000000000000000000000..053accd65874295c38044b79f6abca37193956b5 GIT binary patch literal 1488 zcmV;>1uy!EP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 P00000NkvXXu0mjf>E2o( literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/mail-reply-sender.png b/Media/Themes/Umami/Icon/actions/mail-reply-sender.png new file mode 100644 index 0000000000000000000000000000000000000000..e769da498258b2f43528fcb2c0aa1c6212963bb1 GIT binary patch literal 1181 zcmV;O1Y-M%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00W;%L_t(Y$L*9|OdM4h zhM(D)0bJMuEv1kxG=i3t0LHaxQoSK+sxe-ew9=Yb(^#Wk8IvYP+DIB2Ow*)Itc?bR=v#N$hZLLn^6O1|@~6bJ+Y1cN~U9LGUQi4X$Eagwmq zz7PUMQBYOr=(s>25V(}Oy&%-rHKy_f$?<75aVpa}~s%fc`Wiv1;Y|8R*9&UNtcbMNu#sr}3? znB2CticKXY^bJiAoin-68%pD>X_1r?DJ2u(NqjyZw{G5owyu;T?+nuR^5evezr8N= z7cn&S8JWa@XC<}ehN)fv!#HIg}L1R z$Qk^_`4kp-841rZI(99MlOd2e-?nYzFZC0dj&Nl>%7Nzdyw20X zxpqB<8F#RNnR$~O)lTEADNtHkn#^}U)yAn8cXPgHg!#CQVOp4$z!nY?1&{RL^JU|) zmeI0vZ!B9*?}s8qk=X@wU7@Oc1Kw;6pJwy+$Cof|d3LnD z@!(2`vI;aC$V$n2@m){UdpB)6`|M-&GgrdK2A?;J_OH817~#D|ZM*!@q2~b!ECX@3 zpCORrf-`c_9gptF^9RlxI#54LPJZFZ*3TJjZ`=WxE-*DOxk4HDAhpN@47WHI8T_Pk zY3kP}PriGxFg!Iq>DI@9MYk^Q)-ELAH`@cn{Qxz!uKa=Kr#-5ETH5yTSjS7XfCC6O zuey)w_r>w{v|EB3xV7sDe@4I8xTJZlQ{sLuYFFqT@1a1HT03~!qSaf7zbY(hY za%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0 vW_bWIFflPLFg7hQH&ih)Ix;spF)}MKFgh?W=UH5&00000NkvXXu0mjfkUJEY literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/mail-send-receive.png b/Media/Themes/Umami/Icon/actions/mail-send-receive.png new file mode 100644 index 0000000000000000000000000000000000000000..1bd0d8df90c56714632db94b37d690f3b5d69e04 GIT binary patch literal 1111 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;JpsQj7}P}D;O9UcLw-`xH8ZTD0sEYxi^9w>eVi060hLZE@_=3VVNPQ?jfM&CTb8V zq93l{)h_SRqTtn@nUSL4)vn;);+4OmVg9l1Wv5e`*H2q}I;4Ef%#9b*TDQ#Fd_JRd zvjSMLqI0{VPlu9Yhmv!vvQvk$bGNczm#SZvntPvl^emUmWo{K4+zZ#Zmu+;fSnpZB z(X(Q+SMf^!+O09BX_@_dbF3iYsRwt*$DqsVb{!>aJaI zvZ1cBVbSr%`l{x}nwCY!+FI({TN^ssn>v=A?&)gn@9pUC?VPas?378p3$|Zfv}o?q z#S2!gShnuat&KeEjIq)0gj^z54LNjmMWSoVt4X z^3^Msuikla>-xo8H?QBnclYU&C$C>T`taf1r%#_gfByXC)2Adu|NsAuLTA$$7#OTdg8YIR7@3$^SlQS)IJvm_1q4Mz z#l$5frDPSARn)X~^bL%Rja^(qBcfu`Gc&Vm>!;s*`0&N6SFc~c|MD&TtGyBf0|R4{ zx4X-vX)Ny*7#J9XJzX3_G|nd{EHDe$wyp1+hqtF!M=uvQFCW*H*6!#h(bs*MnmYgf zt=%hJwK?HYS{>ujYoWGxZEf$~jlDa$Ytw>9Zgvg<39&bC-n@D7rtQilnzAbEQga_Y z=A5Ltd0j?!ZvO0$8#iY2h=gx3>*?(L_>ob_FroUBRCLwjMj@+T*RGv2w2L$J^SgHK z-@l!fybhfnD^|}*+H?2uoI4E%_V3)aXWLS98P`LH4<5U|f9KA9Teg|A1TPZj$jaK_ z(bB+p=DDAs1H(iIUM=-#?FE-@WZv5TX>{QCbh~)}z~RG(FA928{5Y+6L{f6=-(K0b zGK~ti5@$GBB9-*}@P-LvDUbW?ChKk%6J1u7R1Zp?Qd*k(H6Tm7x(t!}G|PC7}H5>FVdQ I&MBb@0E2GFTmS$7 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/mail_forward.png b/Media/Themes/Umami/Icon/actions/mail_forward.png new file mode 100644 index 0000000000000000000000000000000000000000..33d884326cc95c6dc1ddcd46c470080420356fd0 GIT binary patch literal 985 zcmY+DeK6a19DqM6o8aOyZPZ!4ZADMV{eJPf*o*99xT$KqRrw*w zSJa`T6KySAZL62n=FFCan|2cPlw(G^_CkY3yhid{zdN^ow&(8o^ZE0sOiGNUIC(e$ z03|+-#RdTPeaZF!Fw?1h3;?8D<_RW1_YG%)`vw5`cZ> zIRK0%fWw9Ifm0saGi z_{~jqsc(f8q~mxcOb(GLme;Yrkg3jLUl2D#{$e55W$>Jl_uibgBZqwDB#HPSVrDJ7cj zLbKgywinIyp@l)TFod3upy$6KgsK81$pG+{n4Opm2OOM2Si&o1SEY@&pDi0#xNhk` z1AtxjXR)knMBn%m0K4PyEN05F@f?mKP+!M%kW(FR3B*43wKZuss+=h%S=B*sTaCQ<0?X1VwzqG z475XO^@kc~^4!C&M;|SZZc{YXd3d_Iw#pk)+b7>6)#|CF16=96V%cJ`)EpI2n4f410-r1910ADdLD>62Sa=E^$p{H!;NizHjDcYdo00hcg0mu? pP;e3G7>);F;X!yr3QlLxBN#aS-(V>t>)!rR@iB?4?>^yN{Rb2{*FyjR literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/mail_new.png b/Media/Themes/Umami/Icon/actions/mail_new.png new file mode 100644 index 0000000000000000000000000000000000000000..f810f7c44a7954fdb654d30952cc9743cf6fffc6 GIT binary patch literal 1102 zcmV-U1hM;xP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/mail_replyall.png b/Media/Themes/Umami/Icon/actions/mail_replyall.png new file mode 100644 index 0000000000000000000000000000000000000000..a0d26a418198f0e03e3e9b1d1cff61d92a989805 GIT binary patch literal 1409 zcmV-{1%CR8P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00e_cL_t(Y$HkRtOjK7G zhDk$hFlpis6XOpPEir90DhQ~Fh*&EqE_7rsn@mAw*!QIf3@UqBgtRh4r3kbX2805 zl49{s!{xYLRVyGr{u@O&Y6aI~dFbx$#>&bH6bc2FmX@%%xVV;ug@p~hGV}BE%E(9!ci-T;ziIqAVWl;1kBCNQOJ6r@;W;^OM8W41wvsFT~9b_1>q5)bbu_7NF-E3 z-~_Tha=CmhWH$=MGQ{F?x&u;#qgD_c8b}A^@|RGVOorvOtj|XbIu-o#?nNZr51!fIvBmfqrl0@Nk%@6c?L{+0&i()qF`nI zQjpF$h5ke%JmH(6=PPGCxWz(C+9?#c?0|frlLFVL;oDjaJs(gJ`1pCfLg2UygSi%H z6A^(nt@%kW)6`#lC{a`##JBCN$x@hU_#ZI$$ z9I#GZ6GUcKpzDVb)VB1hO4zJzXihXl8BY(zq3@vA^$jew^Dxm6gee&xW2KfD$undg zx&umVw^Ml6lb`67{C9}p4402O$pHL66>SulSQJ(*jMz6}9&j6B+08g? zle|HIkO%|#gzi9G+6fvD?p_2KvXE|jf;kpz3<{l4o$XKI)skn(5cr};;4(;8qO zUyF;pN}P=l!8EK4CSm0`9V&*^^;+0OS3ui@_f)Nbguqh@B{<|!f+OA{lvhjPaz2U* z1Y#M?BdaLzOjyMq2^GQO6X9|ozmv^cA4l8^C^#acPAT~NWu@z!L?o}v$zy>OzSH#jQ;q+tegjJX zuxNCqPCozu03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn z^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)}MKFgh?W=UH5& P00000NkvXXu0mjf>E2o( literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/mail_spam.png b/Media/Themes/Umami/Icon/actions/mail_spam.png new file mode 100644 index 0000000000000000000000000000000000000000..053accd65874295c38044b79f6abca37193956b5 GIT binary patch literal 1488 zcmV;>1uy!EP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY3ljhU3ljkVnw%H_00faqL_t(Y$JLcvY!p=( z$A9PS%yhS9ce~IVy+a8#C6r6$Vg=NI`T*#YeZfS$V2n|Gfk#PzC^29wk)WxGm{=e5 zNmFCepd=&)F9BnaTN(=6aw(|EnvMGqf-?rGMUgsgucs z&f2#7Vdvh=y>uSrK)m+qJI?^Ud8I!%cI=bK5#H)oUVdw(ZQD#vrU0m^X`s8iO8|op z1vq$U-(n$kx7N%7tQWVvzWK+We%QQpX`36K6PDxSVb%Ap~3@X)>* zAiaCnp2<6Ua+dgfDfKtcKK=X#AqWZ~2!)D8`(sathWbVWwDdd|tu?ltrKBXxty|aG z{?eO5z#1ua9KmzvxOnye&~qPv-WrR|i{)}TY&(l(Sr~>vKJQRbF`ry6M$(&b6>;(6 zIo7Y=NNMR@dV0DU9=-ukRj4_mM8NXw^pvnHi~70+`~$K{ByKY>(2rplT)K3Ql9Cc` z-n`DbbsGtXqomUlbaZTI*|Ii(z=HsK+qNB8mc`W66emyiVVWkbt!+$9j&uI}pDbCj z94RHOt!;RoODZ)^RaGsKNQAX(Hz35!!#lf+gi;F23bJX_7Oq{pN`L{St+$NPuBBjBK6{|RX`V{^BR}TPTAP=a!03k%QtgIr? z^HsOXW~T!!Ev-UINqzl79LGT^h1Qzd+Is5h8t^=iOeRAznV_+85wTc|tMSXkA#)s;`JUezA0uC5dH^^FV-4Z2ND zi;+?WwASSF4vyn+?297+oH%i+^ZVYuUx6eL2Zn)ZQLqOV5b+`9k3PD*xn;>CE1H`Z zZZ%CK<~X@g(=rohY0000rbVXQnLvL+uWo~o;C}SxgGB7YSAWC6tbz&e> zbY*F7WpWA%Lvm$dbY)~9cWHEJAXI2&AV*0}P>|V+ZvX%QC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn qF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$50000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00Lr3L_t(Y$L*9)XcIvc z#=qU{&TdlLtZ6Ap*pwI=lWHurje;jN&`_jQ(Kgo9Qv5@-Uc`zw#k+SwFCIL3kx~@B zmkLTnENG|)HAS$2(j;q|Y%*~(o`l-PcBA#y-|@}7?|bte^MD<;83_PsB*E6eT~|09 zjw(u1doHJG0H~ED!J@d`@An6L`+8;=SaEgr`#?UQ{|KPA9c|$O>`o*{UcY%e|Kr*7 zr}Xs5sc8Tm0J!aFk^A`Iz}48X*omRRlO)&~v@I;W^Akz<`sMS|CjjMtsooil9vHfm zx}ehVo)=EXTR$61@6UkCI5rIe-9 z@9W8PXBJge4Q|$nKOJygU0v#-!w2ss&nA1i_jmKT{0eF{3R#3;$-s;A{)tXN)!9zxTi< z2%=?!^A{)3*4Ad4smZ|W^@1_RZrq$Ib6anM^*DS!pDAcEa5|l(;Y6apPH98x0N`@D zOl!j=!WiQLG{M3m+knAwT&dBbcKHAN1S{srCv~Sc5C8xGC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjf?TLK` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/media-playback-pause.png b/Media/Themes/Umami/Icon/actions/media-playback-pause.png new file mode 100644 index 0000000000000000000000000000000000000000..378ab5edb36c8fb6d2bcd0334c988c8d96239c6c GIT binary patch literal 798 zcmV+(1L6FMP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JOML_t(Y$L&UPt#SW>YS>YZs5lLd!wy$A3$7#{Q=+{z!*TQ2J;=j*RsPYAA}IW;nDq5DdnBA zUrPDs^!qW$GN!4U-`>5Q-DBo@0MuS^L z{xt%zMZk5-YHH2yCV>?F_MG+Uuq;w#pw4L}tzyJ`(vF%{FTH^QOCuqY^j_o99x7!wQ zd2s<_mcWLfAWxFSs&u48L%BA;9IGR+03isDqpZ3pjPkMwBA}rfP)gYmb-UgD1pR)W zi`^;JfO1sP0NU-fy1`&z$EZ{Tj?=IrdcEHM5eerWTo!sekc!Md8NWm#C zBoRiV(f+?^JRVy>93>TCx7*DDU^1Cll}XY@UH(pj>2#V4A?8n>J&MYHrBr_V;NTJf z(lq5f&wo9B`mpjA#c{Gq)AZViDmR6hTUnNQ#+ZuVbzSy6&k)gqnOnvfFUztD#`8RO zU3W!9OWfE$Wq$!yIhrA0r1(An001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHY< zIx{soF*YkOFgh?Wmh(6<0000bbVXQnWMOn=I&E)cX=Zr004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00Ux4L_t(Y$JNx^YaCS^ z2k`Ij%$c3NZ?>h$?!>0aY<8Piuh_ySDL$pF+tecDLTcMTpvH$HtuMs~LGTu^Ao^l8 z)h2>}MB)t*vHI2&Vq4OUW=$_VEf*E&wUkDBNK_;iQaOlay6IBRRQP# z>_-AT@zkCZpPrjh0s&tlm(LyT+O?CeFK058i@=k6PE1Z5;hGj>=@-(jLaDGf z6bwwZZ4a$xmREB1mWeF^W258z*6kbMoMUKUKOY<#@Ml(54s`G8dR}5!T3fqm0Wj6jWD9p}2x zAu$FfF)+qJ1em6QYPE{~zQ?7J^g*RiEG9iZWwd=q$2HD*K~+^pRaGOK&29`NlgT!h zBELE|I?jb5z?cNaSR;uDRogM>G<}$jBkLX&HmTpf{pvTBA+;ULY2W z**9+0K97Nc7y}~$5dl<}9S5@Pg2(N}YIYT8rq5Jzchb;jBNfKl(0|4e1=8Z2dd|vVS{BOQ?^7O3N>&?gG@eK!K6VTn=-2iOc1`$D) zCAef4mVR5Zr>EZ&k|bR?_VV#}QmND$5tY#*ngB&n8f(TFd|p3t`8>{^{h*S~{_*p( z&piG5)YQq#0E$Gk>G0#NHWil#vvaeRg|8QGMmsuAo%>{F9zcPJ%*_gK0yy~Q+XZXp z{CVB)_r3l8yVIYwx3}kssPdqNTLOyWzBKpc+~VZKk+;(6^q&vfj*Wtv-raF_pmqUG z0C$MUdYHoZ4aad@j^ij>uk;_q-%^7?TS@`r?f?J)C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML zHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* XE^l&Yo9;Xs00000NkvXXu0mjfMiBdA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/media-playback-stop.png b/Media/Themes/Umami/Icon/actions/media-playback-stop.png new file mode 100644 index 0000000000000000000000000000000000000000..b95ff2480f434e816ee681d819be331ee1710697 GIT binary patch literal 681 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s79PZ!4!kK=EH?|Lx>3j9kuXZbp9*P`xv zm5CR+OT=Bu1uwpK{lX=>*Jq|nMY_j=w?5*lS3g_EuDGi(udJ->+i&aL2iE=DopCyS z|I}YiCX6B56eJj5==0^=3zKN8PJh1q`{Gp5_i$uwjhgl}`@dwT$E@dG3tm3hP|U)Ru{G+}>#t_QCQ`g76*?TA zl;10qIelL8x$d>srFVI@-}@QG9w_qic>;^l0SWGaO?wvX`e1u*w@u=Hp*{Tf7?gU{ zlWSDt14LK6`qH&VeCG0?wPAmiUX|@;Vqjr1K5TG!L#wISHH(PshijTQGctuo2t9kM z(D%4xV#5Jej~>RfRSUWOF4ZY2eN^H&9FWe}w4g!7)5u|FW4GV?=QHlL+jMg<%n1Iv zXv(V4My>^Onl_&aepg~u$LF;)Xqk$Z`-inj8(%zM5TmCpUAJ@BKAUf>>5t+z-;61o z@Vk4}E4zk6Zdb0(61aF=?ETwU{I8{p{x4FvmJ_GJz`&qd;u=wsl30>zm0XmXSdz+M zWME{VYhbBsU>0I%VP$M;WoV{tU|?lnFzvI2A&Q3F{FKbJN(LhXLqlByGhIXT5JMv? zBXcW5BZ!9Qkuggc7#Ji$HU#IVm6RtIr81P4m+NKbWfvzW7NqLs7p2dBXCuYHz`)?? L>gTe~DWM4fwTTSX literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/media-record.png b/Media/Themes/Umami/Icon/actions/media-record.png new file mode 100644 index 0000000000000000000000000000000000000000..d13d3ef8b8002b4faa73d1b26db642568e6aefe2 GIT binary patch literal 1258 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YNKL_t(Y$L*C*Xd6cq z#=kc!X=SZewpVr}Ir2Znq$w?N6(u+n(-dNygJ^pQJ>*!DQ%Kt!n$T0wA%@<1XbJ7L zp{3AEO9(l{g%Sv{n^47R>_do+k!nkdEIIXRrPZ!>rUzTZ|CG8t=YtP+hT-?U$2?|% zSNa#x9?nPr0sy=Kyec5;KplX&;WYrx{|)e~K+m_phY_L^K44f9qT3;QRs&}K65_Y> zz)x#H9zf+k1t^LV`V=_)UbXslcOuc>l}yT>zCO6Ry3i<`QgCDi{b7wXIz|f*yS?J#YH&PDwy97ZEVcde&|qF z^xnPG2N?U~Ni@2=y1H6_X+R>8a32O@r*pX<4_>*_>FMt$4>Bg-fcc1~>C|1Ox&BuVUNp2gk=<)rSv(YIRq*sMi6@ z0%|qrlamMz54-)A^+7ls_O=X2rBbeUGnt6HypEcD;JF;Y7zAS=#y~{t76Obh z2q7QQxptij1 zSxF{m0a&f>fYa#c-$ICjT_`|~#Xv;Bwvf$m5J8T`U>6F2BrPi=BYG;8YCWc!rd6C+ zY+~u=O;brEz!YU`7aMM_B}GBokt0~Tb<4B{2fxrXZOfCf%ipS$QA`-2p2t*`=5XHI6 zmp^+pG4XvZpC9Z=CcV5`1uvB#DittQ1q%j2J|A*7ZkT!|v!GnPdg--zd}T9y*AAM> zX0vTwlGOd=`0)#(Q21CK8WKt*B6~EAS>-aX=5qD5r%#DOp)U_kO?|WCIK|%H-rA0W z@e&CkBmkYoYuBQs>(}41GMP9kl{c8GK6dr?rnR$Y)BDbyTLw@jBEB1AFF%}SkPN`1 z>$*!-)yBps)``g33*(hq>L1=SsHBN#{u}@R03~!qSaf7zbY(hYa%Ew3WdJfTGB7PL zIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQ zH&ih)Ix;spF)}MKFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@W UXPfRk8UO$Q07*qoM6N<$f&eigfB*mh literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/media-seek-backward.png b/Media/Themes/Umami/Icon/actions/media-seek-backward.png new file mode 100644 index 0000000000000000000000000000000000000000..134fc02c9598f273f7f29b75789065e12647ea05 GIT binary patch literal 1184 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/media-skip-backward.png b/Media/Themes/Umami/Icon/actions/media-skip-backward.png new file mode 100644 index 0000000000000000000000000000000000000000..070ce76f4b8a0ec02ff44a520bcfffe63f4eb3d8 GIT binary patch literal 1246 zcmV<41R?v0P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00X;8L_t(Y$L&*1Ok7nI z{_dOi?)#ZxW|)zIN<_w?I)Gg?GXb?Jg|--qt);2s6qFz9UkK1itU{y)B3dOH7lwqA zrpARU6Jx8i$_h;kv5UGWi5sISCWN%1fmayld+=tCibIx~w z`}n`%u$>tI;olPQDvDC-bZSM6v3$$2EC9yeym1032A~00z6-Ab*am+US3}5NhX&VW6Yi-BjELV2>>LC0{|W@E>>#x>`s(E5`6Ad|EZKO z-|y3JEg+pv!{y8a0PNjg`+Zw$+j>Joqh^|hdtqUI)4*g|hH07^fr`q?6k`mMD1kA? z9^X-Mp{J+Eda`<#`?j&lmX>q~f&f9tm@X2Dpu6jYM-&~dm6c@>5d?<=>2!KSAR3LP zrlzJczAq1lQ}zlZF1d<|wh6guX4*tW5%uNFJEUg+dkph}j|6*VoT|`2KLmjq7v$?H#SlSAMzjyQ*ku&6eQWA2Y8Xf2IAm z^B2CJGEIZx%`t0w`f3RgiA3a`oSZCcf9b`~ySh6U z7WKtBBBEIQC=pRW*L7ba(fw+?Id<)bA1D8`ENjiq-(gdG4*{eYV>%HTb#-;U0|NtJ zb#}h?NyCB0B!KN%_bm|77c(<67l%Lk;P|=E&Lt9wL=r%7lTCd-za@&I-qz9T=bY<4 zpWoWpHpUDhGKYqSewv+~-T&^;V3#O~dvcqc<*$#8p8a%UV&coOu`@lQC{@|F6-1-~ zaAjixkYtRlVhe~!0^rVmEX5eRxrKKC$O9k)NHfOf?Ia^2fruFH0U}a~i2nmG5s`SW zJooY6;4jwVU^;i$+rt0=03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`If zHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)}MK zFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@WXPfRk8UO$Q07*qo IM6N<$f~q?q`Tzg` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/media-skip-forward.png b/Media/Themes/Umami/Icon/actions/media-skip-forward.png new file mode 100644 index 0000000000000000000000000000000000000000..56e943d7d5cfd75cff894c5d7f2090f1fe7ea78c GIT binary patch literal 1318 zcmY+7dpr{e9Ke4=w!Ar?i*#|ushGA|nng7x^YJLJ@~W`LX_&M&L}qDBrSnXv594ml z=t^F-JW`ELlDQr#EV-0d;XV#GcAxJ4y3hCX{qy_z3J&;s=xS}$0svi4D#afFOumZJ z1ONrcWeET@V#q#ZfLoWem67WJz=B5g_W?-#0f3tYumS+wj{v8z0Fw~_q*8zli4}MK zoB=>H%FBZSs;^YGi3$N=eIhmFBmhwVb>CygZU7B+PYOAZ+C38(=Dmq-;xZb}9=dv% z&A0CCdTL*K#Q(C9v8yCL6jRjrBI%RcdpC)UP%NjI*J(=0jXt6E2Wy@g?A=B<(g2wI}NFJwB_$Tgd9PESwApD|xFx?b9DlKv^BY&>n@>StAZq59iymG0Wn zSr2F}*3g0;5W<+PvPLZta_T!p?;3~Vj(8Sm={)%98v(_5ZxIr*(b5`r_hO4y_;R!O z@zfx&U|%ZA-S79Z*ONPqWAuJudiM`MQisoMY5Mb~OBvGWzR3rSbcWflt7$WME+FCQ z+iyO#mPfI&AuW2u9U=)}|u%&>Hd-6W|EZ7)K0?%GrkIKIu5 z$u!rO&CB0Bp7zQwrb;?6s90cKFt6js?opp)PE~ zb}~a%(RQoq9o3Dtx-6H=FM<{VbQG<0Ore_4!YV~p>A`w3e5bFte*3~T^VWzr9n8@+ zc>@w@YN|%lO3$2E-lAUEybHVdtzBaw6mb2R`3ift_GuE>jqhC0z)G_SKf0)M3{_H~!IWt0$|FkCR= zbgJxjuyxKo)NLp2yEl-bM;FWrUy9moS$9?@Ht@^^Jd|bV8c(TAJG&(%<@S&KV@+o# z&OK8okQtlfNM8o~${n(ad0|$T!5E{420Oukq`G4dM{#9WDk1gb!@0uk$!*ubkgRw_ zw>2u*efgqTT%(?RBqwcU$i@MUzD(fc|NZLAIj=)}KL70=<&q0Tu-S|JQC02P%=CfD zjMc{eiE^YtbFf_kC|buWYil=}A|M!LpL3{V^J=xIrF)F^q5VyI^Muf@cR{?Ah*U$I zj5*XjzO=LyKD06~3JVQAgD|iXqG)!5-Pl+7axGBI8E5nE_h*>36P17TZf}zZ`1v_$ z??qc`0*yxd9#iMztt8EzILcIUtvtz4MbPg|yyHy9D0R!q^t+;(d_~n+{%ieS=~H5H z$%iNYyEP*+bJo61zJN0WMs42DlYbJnVr36AsGeBalM6WQS%l04Cu=?Ze7HN~dsTay{{<5v;WYqY`t=aNjAn5nljy+VaO`6f7$<3w YiFA8rQcU@*^F{ywPj_F+4c7?XKM-$2EdT%j literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/next.png b/Media/Themes/Umami/Icon/actions/next.png new file mode 100644 index 0000000000000000000000000000000000000000..c26fe61e751a7e45a068cdaf5acb23914578d810 GIT binary patch literal 1100 zcmV-S1he~zP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00SvWL_t(Y$L*9&Xk1kk z$A9PDnWDXv;VNxc74 z;?afy2Y0{67eDqV$sG=hu?Fib)@h8@7^k_X?Ll1Raq`pS3+15nT<%RX(9jHw)r7iC zsLMnqAk+b|35iWWqyr)o5a}}C{&t3IW}0Uo+jGmRTi5lqcd4$10NTcs;}Wxr(_Af0 zGBf`xbLBq>bcvV%BSNf*up(mYYL1EGIi7yxh1Lyicf6MEmPeNzGC(PqWwLk?hpXJ9 z2h#VDRv>*zMrH7nL|B0p2B*TxDi8>9dtE?m!yA(OO&%&!k*5Tn)PTlX$K9*jS)aL= zkItM}oSQEm8)-Fr>jKgyeGln-s3z|^$WsDOcx+y?6S=sOzJcS5^K%OazC3E)tvz20 z@T;57Ee8;Qr$hy!1X2mkP9LGQrG@)%dz_iVoI7>8XJPi|@Wt=??J&@qTgJsJlXz@`BbkomXt zW}qoib9LX|rADlAY+L;dx#3*&&4sUi8v9W1y;k5)AOqMWsR49Sa8!2&X$#5$AAdFw zjgOD#&Y!Tan~;TN!fovprRg$JMCEID0R5+XqhBsh9lP+p?M-6iL{V}sEmI57$tqG6 zTyNLJ?&}{$3q4~e?QjCLsbp1143vSOTGWz%QR4HHyZ4hKsvU_`N7X@AiwG0x+GNX_ zOkRdesoGt7sHR%+e{O#REe&ow2E$w?0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGP zFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG# zF)}(bH##vgD=;uRFfiv?T%-U102y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw S?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00Lr3L_t(Y$L*9)XcIvc z#=qU{&TdlLtZ6Ap*pwI=lWHurje;jN&`_jQ(Kgo9Qv5@-Uc`zw#k+SwFCIL3kx~@B zmkLTnENG|)HAS$2(j;q|Y%*~(o`l-PcBA#y-|@}7?|bte^MD<;83_PsB*E6eT~|09 zjw(u1doHJG0H~ED!J@d`@An6L`+8;=SaEgr`#?UQ{|KPA9c|$O>`o*{UcY%e|Kr*7 zr}Xs5sc8Tm0J!aFk^A`Iz}48X*omRRlO)&~v@I;W^Akz<`sMS|CjjMtsooil9vHfm zx}ehVo)=EXTR$61@6UkCI5rIe-9 z@9W8PXBJge4Q|$nKOJygU0v#-!w2ss&nA1i_jmKT{0eF{3R#3;$-s;A{)tXN)!9zxTi< z2%=?!^A{)3*4Ad4smZ|W^@1_RZrq$Ib6anM^*DS!pDAcEa5|l(;Y6apPH98x0N`@D zOl!j=!WiQLG{M3m+knAwT&dBbcKHAN1S{srCv~Sc5C8xGC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjf?TLK` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/player_end.png b/Media/Themes/Umami/Icon/actions/player_end.png new file mode 100644 index 0000000000000000000000000000000000000000..56e943d7d5cfd75cff894c5d7f2090f1fe7ea78c GIT binary patch literal 1318 zcmY+7dpr{e9Ke4=w!Ar?i*#|ushGA|nng7x^YJLJ@~W`LX_&M&L}qDBrSnXv594ml z=t^F-JW`ELlDQr#EV-0d;XV#GcAxJ4y3hCX{qy_z3J&;s=xS}$0svi4D#afFOumZJ z1ONrcWeET@V#q#ZfLoWem67WJz=B5g_W?-#0f3tYumS+wj{v8z0Fw~_q*8zli4}MK zoB=>H%FBZSs;^YGi3$N=eIhmFBmhwVb>CygZU7B+PYOAZ+C38(=Dmq-;xZb}9=dv% z&A0CCdTL*K#Q(C9v8yCL6jRjrBI%RcdpC)UP%NjI*J(=0jXt6E2Wy@g?A=B<(g2wI}NFJwB_$Tgd9PESwApD|xFx?b9DlKv^BY&>n@>StAZq59iymG0Wn zSr2F}*3g0;5W<+PvPLZta_T!p?;3~Vj(8Sm={)%98v(_5ZxIr*(b5`r_hO4y_;R!O z@zfx&U|%ZA-S79Z*ONPqWAuJudiM`MQisoMY5Mb~OBvGWzR3rSbcWflt7$WME+FCQ z+iyO#mPfI&AuW2u9U=)}|u%&>Hd-6W|EZ7)K0?%GrkIKIu5 z$u!rO&CB0Bp7zQwrb;?6s90cKFt6js?opp)PE~ zb}~a%(RQoq9o3Dtx-6H=FM<{VbQG<0Ore_4!YV~p>A`w3e5bFte*3~T^VWzr9n8@+ zc>@w@YN|%lO3$2E-lAUEybHVdtzBaw6mb2R`3ift_GuE>jqhC0z)G_SKf0)M3{_H~!IWt0$|FkCR= zbgJxjuyxKo)NLp2yEl-bM;FWrUy9moS$9?@Ht@^^Jd|bV8c(TAJG&(%<@S&KV@+o# z&OK8okQtlfNM8o~${n(ad0|$T!5E{420Oukq`G4dM{#9WDk1gb!@0uk$!*ubkgRw_ zw>2u*efgqTT%(?RBqwcU$i@MUzD(fc|NZLAIj=)}KL70=<&q0Tu-S|JQC02P%=CfD zjMc{eiE^YtbFf_kC|buWYil=}A|M!LpL3{V^J=xIrF)F^q5VyI^Muf@cR{?Ah*U$I zj5*XjzO=LyKD06~3JVQAgD|iXqG)!5-Pl+7axGBI8E5nE_h*>36P17TZf}zZ`1v_$ z??qc`0*yxd9#iMztt8EzILcIUtvtz4MbPg|yyHy9D0R!q^t+;(d_~n+{%ieS=~H5H z$%iNYyEP*+bJo61zJN0WMs42DlYbJnVr36AsGeBalM6WQS%l04Cu=?Ze7HN~dsTay{{<5v;WYqY`t=aNjAn5nljy+VaO`6f7$<3w YiFA8rQcU@*^F{ywPj_F+4c7?XKM-$2EdT%j literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/player_fwd.png b/Media/Themes/Umami/Icon/actions/player_fwd.png new file mode 100644 index 0000000000000000000000000000000000000000..6212e20ed5c18931b307941f1438e05df798a6f8 GIT binary patch literal 1229 zcmV;;1Ty=HP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00XK?L_t(Y$L*8dPg`{u z$G^YdIj22+VTHmd(pSu)61Vm3fK)>?v2A9Qwlbrc+q_v~vI~NVE?acb5Uvae#p#V% z+%9~{BF+oL)GcIA{{)5!Dlb7OEoCJwU3>a+&d&=AW+1^`nV3B{zvtrnJfH9LOTNH^ z`rj8I~?xL*&&{FG5_Y1JT{K`vXEiIPj`T2!~+5&Z(w(026*y{%#InbNS z=4SKx{N0+Ne+6hf>^bz2Px5tJEc+vRqhTqrlvo2G17MpMJQ3&|^msiXQ769kP^+af zH#@&j(3gm&;}1@w$(N zf?*3U@aXP-TIlTPG%qbJzF@UmeF{PRjPFPx-6ZIyO6bgndRkZ@SSb)K3Bt6eZg|;@AE4h+5Zg<-1E9L4O$8mD8 zSS)V=y1Ked=EjEpXf%dOr3{Yapr{HIr3$axCj^5bLpr_sOfH)>heBZu2vDt7Q7RRY zzng=@?xeA3ub|iK4sYbMC#`nts?+XToSmI5Pymo5NdW)?0+p#y6a|V>LAhK)K`y}D zV8*M*j_W~)o)8K`C@7^6G#UsR4N7Gh9OV!W9uXZ5m(gG_JSRyK1ppoZ{C>YOdGp3! zf5!3PC#W*7vdF4 z>4*J&Z^Sz~I@SOfK?nc_ov1@5lVKzA5qTw<_|fC{zSDoY{})2Yj)e1D9Rrf7B#XyK z#nsnry&e)DzV)=#&Vtv35xmo8qO1W+J^?8v}90J$WW z&z?K;yVKS7=BF1g{rpcus>-s1g9ABJy?Nlm*vFSmCR3IWvTKU}HqY4p@$vD;*4Nh? z7-QTnntfAKQ}&682`6Jrv**%n!#rb5*h51ZV_L=-xv%99>OZeP0p5CR)!r$^&Hw-a zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuG rZ)S9NVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjfT^Sc< literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/player_pause.png b/Media/Themes/Umami/Icon/actions/player_pause.png new file mode 100644 index 0000000000000000000000000000000000000000..378ab5edb36c8fb6d2bcd0334c988c8d96239c6c GIT binary patch literal 798 zcmV+(1L6FMP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JOML_t(Y$L&UPt#SW>YS>YZs5lLd!wy$A3$7#{Q=+{z!*TQ2J;=j*RsPYAA}IW;nDq5DdnBA zUrPDs^!qW$GN!4U-`>5Q-DBo@0MuS^L z{xt%zMZk5-YHH2yCV>?F_MG+Uuq;w#pw4L}tzyJ`(vF%{FTH^QOCuqY^j_o99x7!wQ zd2s<_mcWLfAWxFSs&u48L%BA;9IGR+03isDqpZ3pjPkMwBA}rfP)gYmb-UgD1pR)W zi`^;JfO1sP0NU-fy1`&z$EZ{Tj?=IrdcEHM5eerWTo!sekc!Md8NWm#C zBoRiV(f+?^JRVy>93>TCx7*DDU^1Cll}XY@UH(pj>2#V4A?8n>J&MYHrBr_V;NTJf z(lq5f&wo9B`mpjA#c{Gq)AZViDmR6hTUnNQ#+ZuVbzSy6&k)gqnOnvfFUztD#`8RO zU3W!9OWfE$Wq$!yIhrA0r1(An001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHY< zIx{soF*YkOFgh?Wmh(6<0000bbVXQnWMOn=I&E)cX=Zr004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00Ux4L_t(Y$JNx^YaCS^ z2k`Ij%$c3NZ?>h$?!>0aY<8Piuh_ySDL$pF+tecDLTcMTpvH$HtuMs~LGTu^Ao^l8 z)h2>}MB)t*vHI2&Vq4OUW=$_VEf*E&wUkDBNK_;iQaOlay6IBRRQP# z>_-AT@zkCZpPrjh0s&tlm(LyT+O?CeFK058i@=k6PE1Z5;hGj>=@-(jLaDGf z6bwwZZ4a$xmREB1mWeF^W258z*6kbMoMUKUKOY<#@Ml(54s`G8dR}5!T3fqm0Wj6jWD9p}2x zAu$FfF)+qJ1em6QYPE{~zQ?7J^g*RiEG9iZWwd=q$2HD*K~+^pRaGOK&29`NlgT!h zBELE|I?jb5z?cNaSR;uDRogM>G<}$jBkLX&HmTpf{pvTBA+;ULY2W z**9+0K97Nc7y}~$5dl<}9S5@Pg2(N}YIYT8rq5Jzchb;jBNfKl(0|4e1=8Z2dd|vVS{BOQ?^7O3N>&?gG@eK!K6VTn=-2iOc1`$D) zCAef4mVR5Zr>EZ&k|bR?_VV#}QmND$5tY#*ngB&n8f(TFd|p3t`8>{^{h*S~{_*p( z&piG5)YQq#0E$Gk>G0#NHWil#vvaeRg|8QGMmsuAo%>{F9zcPJ%*_gK0yy~Q+XZXp z{CVB)_r3l8yVIYwx3}kssPdqNTLOyWzBKpc+~VZKk+;(6^q&vfj*Wtv-raF_pmqUG z0C$MUdYHoZ4aad@j^ij>uk;_q-%^7?TS@`r?f?J)C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML zHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* XE^l&Yo9;Xs00000NkvXXu0mjfMiBdA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/player_record.png b/Media/Themes/Umami/Icon/actions/player_record.png new file mode 100644 index 0000000000000000000000000000000000000000..d13d3ef8b8002b4faa73d1b26db642568e6aefe2 GIT binary patch literal 1258 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YNKL_t(Y$L*C*Xd6cq z#=kc!X=SZewpVr}Ir2Znq$w?N6(u+n(-dNygJ^pQJ>*!DQ%Kt!n$T0wA%@<1XbJ7L zp{3AEO9(l{g%Sv{n^47R>_do+k!nkdEIIXRrPZ!>rUzTZ|CG8t=YtP+hT-?U$2?|% zSNa#x9?nPr0sy=Kyec5;KplX&;WYrx{|)e~K+m_phY_L^K44f9qT3;QRs&}K65_Y> zz)x#H9zf+k1t^LV`V=_)UbXslcOuc>l}yT>zCO6Ry3i<`QgCDi{b7wXIz|f*yS?J#YH&PDwy97ZEVcde&|qF z^xnPG2N?U~Ni@2=y1H6_X+R>8a32O@r*pX<4_>*_>FMt$4>Bg-fcc1~>C|1Ox&BuVUNp2gk=<)rSv(YIRq*sMi6@ z0%|qrlamMz54-)A^+7ls_O=X2rBbeUGnt6HypEcD;JF;Y7zAS=#y~{t76Obh z2q7QQxptij1 zSxF{m0a&f>fYa#c-$ICjT_`|~#Xv;Bwvf$m5J8T`U>6F2BrPi=BYG;8YCWc!rd6C+ zY+~u=O;brEz!YU`7aMM_B}GBokt0~Tb<4B{2fxrXZOfCf%ipS$QA`-2p2t*`=5XHI6 zmp^+pG4XvZpC9Z=CcV5`1uvB#DittQ1q%j2J|A*7ZkT!|v!GnPdg--zd}T9y*AAM> zX0vTwlGOd=`0)#(Q21CK8WKt*B6~EAS>-aX=5qD5r%#DOp)U_kO?|WCIK|%H-rA0W z@e&CkBmkYoYuBQs>(}41GMP9kl{c8GK6dr?rnR$Y)BDbyTLw@jBEB1AFF%}SkPN`1 z>$*!-)yBps)``g33*(hq>L1=SsHBN#{u}@R03~!qSaf7zbY(hYa%Ew3WdJfTGB7PL zIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQ zH&ih)Ix;spF)}MKFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@W UXPfRk8UO$Q07*qoM6N<$f&eigfB*mh literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/player_rew.png b/Media/Themes/Umami/Icon/actions/player_rew.png new file mode 100644 index 0000000000000000000000000000000000000000..134fc02c9598f273f7f29b75789065e12647ea05 GIT binary patch literal 1184 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 IM6N<$f~q?q`Tzg` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/player_stop.png b/Media/Themes/Umami/Icon/actions/player_stop.png new file mode 100644 index 0000000000000000000000000000000000000000..b95ff2480f434e816ee681d819be331ee1710697 GIT binary patch literal 681 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s79PZ!4!kK=EH?|Lx>3j9kuXZbp9*P`xv zm5CR+OT=Bu1uwpK{lX=>*Jq|nMY_j=w?5*lS3g_EuDGi(udJ->+i&aL2iE=DopCyS z|I}YiCX6B56eJj5==0^=3zKN8PJh1q`{Gp5_i$uwjhgl}`@dwT$E@dG3tm3hP|U)Ru{G+}>#t_QCQ`g76*?TA zl;10qIelL8x$d>srFVI@-}@QG9w_qic>;^l0SWGaO?wvX`e1u*w@u=Hp*{Tf7?gU{ zlWSDt14LK6`qH&VeCG0?wPAmiUX|@;Vqjr1K5TG!L#wISHH(PshijTQGctuo2t9kM z(D%4xV#5Jej~>RfRSUWOF4ZY2eN^H&9FWe}w4g!7)5u|FW4GV?=QHlL+jMg<%n1Iv zXv(V4My>^Onl_&aepg~u$LF;)Xqk$Z`-inj8(%zM5TmCpUAJ@BKAUf>>5t+z-;61o z@Vk4}E4zk6Zdb0(61aF=?ETwU{I8{p{x4FvmJ_GJz`&qd;u=wsl30>zm0XmXSdz+M zWME{VYhbBsU>0I%VP$M;WoV{tU|?lnFzvI2A&Q3F{FKbJN(LhXLqlByGhIXT5JMv? zBXcW5BZ!9Qkuggc7#Ji$HU#IVm6RtIr81P4m+NKbWfvzW7NqLs7p2dBXCuYHz`)?? L>gTe~DWM4fwTTSX literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/previous.png b/Media/Themes/Umami/Icon/actions/previous.png new file mode 100644 index 0000000000000000000000000000000000000000..90ecc8b397791e29e3ed1c811a076983119527e5 GIT binary patch literal 1135 zcmV-#1d#iQP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00T@(L_t(Y$L*C(XkA4Z zhM)O4=QKB|V%iFct(byFM5uzs~OWzN9N`+nd1&6$D! zqe+OhZs35+7M!|LtZwrqR- zg~3A^CAD2`QcbgXaG2I&QIGfBu_JeX&*SSn?XStwYYOzgL}$tBZwBt&({*=upEoo2 z6}`6%A~n@;Ntd$+8`lfTqStCV)4lmuf$WHPmlW#c;Ld$FbZpwJCTG6@9FdK9dH*{t zQb+^>A(mZ!W$gKOFdCW&^bM)q89(#UqdT5$lYwJ$_H(3ADCzO_nWIRdkwT)RLLd=B zklXs?>MY6*>!Iu0+F#u>us?I5Jj<_Vr%}>FOAqTT)n0 z^pB{+9UC@3w=4HV=FEj3nY;YUGO>&BgvF{VRbhx+4IF5E0<|9FYB=YbR3Pa@A{-Gl zQz0}ZYC_>~C?CKeaY!W)vYxB64p)Jay>wNJ8vcVHTB4`s&!0Q;>BI-cTQ~I3edB$E zri7IhP*rpQev2Z=DAu(qg4X)C*CMf^+FXE9?5Q`*(8a||Bgao1Ew%ew=8^`M}HQ6XK!RC4i~Nzy8CQl;6JBy6ITst~yX0Kyugg=zPl z+*qidyZGC-N?6njGtro-P>#K>rhHH4)s+Gka4bJU$vmQdx0v&%-Hf=d__M#>b<9t> zW41;)WlV88$!Us5o$8N>;ZovB0*Q!ONp`JJu~jOCq&k}th?0!|^Y{xt zPe=(CsL}oa001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000XU000XU0RWnu7ytkO24YJ`L;wH)0002_L%V+f00d!4L_t(Y$Hi7_Op{j_ zK5haMVv5G7KT!GM0u)2MfdYyVZ>V@zRFpZHR z%jPatq!fy5Z9xQFTW~}u9h94z`c>WK*>gUtS!SuuA6t_rUwXdtKF@o3&S{1Czd!E_ zF@c!q2QNG~d7cnQbA`Cir{W8IcaP7{C^F)C$UpolDaqqLaRQ#QGI&m(hNrX?o{|zM zr%pjRejImlb5Wtydbsz~ebhH#X^Q`2>=i}oQ_sNqgdHeeHXkWMPp-5^f>?>BlzHAw6ixq^8u>@(Vq+DVxntixv~Ab`)3$B6N|N@qYt?tYgFlDx8_THG8iCC@TwjT3u{P}2N_KnQm5*G*4tXUY?v&WCXv172YWa)9+k|iq46se}0y02fy zKz~1OQDBf2)UkC9axODw;HPQRP#+P2D^XDxV2g4C0)`RD&xe(^lm3=MW(w#zhi>*? z4%AF5*3nV7*rwYQ(XragF)^qT0+oc0aB+4#)xHJXd3mt1Y->^ynkb~Yg_-wsbf9kU zUeuECLSP`S(8h|(g}Y8CzU5#qc*(2Ak4JBMx^Dq#(;@cHrq!Y`G!&+g5LC+(TV4)5 zopW*0Bp4Z3LtG3CLp}F8E?q(;MSqZMgn@fphYtA`;Ni&mWZXDqfY!5yhr`58CT65( zo2sTye@t#5^u*V!#F&=ih4jU%W$XptutfCDt!!WWk5Xx>xh4n zU;bOqCApCaIe4 zsW<9TA+pOf8l{*1Z(?6-XgdR;C#pEBNID=lc%thb#(Dyx@(XYH=om;k`n``A7&4bs4(D8ROj&b3_UwiloAtH%nA|p+R zZ{8E)H{$nqh4_Q`^BuNDKIM10{<09w-1|0>Nu+q=1N@1M@Cr>NB8UVcm3W1Cjd+8| zBGSEtGV_;+2}IC;{%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG# hF)}(bH##vgD=;uRFfiv?T%-U1002ovPDHLkV1jirW&{8L literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/redhat-home.png b/Media/Themes/Umami/Icon/actions/redhat-home.png new file mode 100644 index 0000000000000000000000000000000000000000..1d654650541eace316d850109a8f8ec79e084a20 GIT binary patch literal 1072 zcmV-01kd}4P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/search.png b/Media/Themes/Umami/Icon/actions/search.png new file mode 100644 index 0000000000000000000000000000000000000000..d543502ad037a7c4d11cf1f6b0468f2ed08d6e12 GIT binary patch literal 1414 zcmV;11$p|3P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 T*0Dx800000NkvXXu0mjfL&DnB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_bottom.png b/Media/Themes/Umami/Icon/actions/stock_bottom.png new file mode 100644 index 0000000000000000000000000000000000000000..ecea1ee672ad6e5b4b07e8b0e716840f20eb5dd2 GIT binary patch literal 1111 zcmY*YYfO^|6uoHGQVNO_1)K~AIEIw81q*yMFt7+k=y)_dTLzSZygT@yz@U_>3k;a4 zTN(zm&==4`c|YYfBsw0Jmka^79~hPyHX8{0{>Koz<(0#*XH}Mbv*m45HLja1;0SK=FyaoW_V}N`j zz>@?3q!!>5ui;i06#$HRfiZai5co!yW-`11jD8LcX8HpFl81qJf#hNSr&m4(K{3<< z43%&2Z%Z%qjV~~?dvV`Sw=@nUDM0?d_KB)7#oPwS1g z&`LDn|BXd4m7}rZkyIoP!>BtP)_5dBj%6y0nJR3g%si?U7p=iXYjLsdCUJ7hxK{JH zPCQFt&FaLnbw}A2Fz6T%QYx-ID@dTY^;-FcQ+Ae9>jMHq-xpg$T*@3J5kUMrP zwbdzg()`M8^E8cfnud@zWuC5e$((*KtDT&qbIa)>@pP2Wf3V5zb49zTXb*x8y5&!O zoUi|&pv#-z{Yg=eZ_$9y7rp)^eE}tto+S(COZ(}i1D8sN{RI6%SBJu{-lJB{ek#^S z-CT(Jb|&e&>FkDSo>ZSD)$?R``SSG)#Z0MkR-l{{D3=SAKV_=s1gg0*)tpduPoUBZ z)Ovw>UZ9>YYhKH5SuAT=Dr+@R*t#s#tOzw5h1&TF?P_`3YI)m(^7f^Q_O|2oxtu~CTHI1z)$JQ0&8>)#7^~6T=146k&=gdV@=pejN;6Si7J1K0%LdQ((`MRS7{r^Iul33(X3@GvWZCSH81acy z!6r9uuT-z>5Rce#>@N0FJ#B_!s&}2n(!`Mx$^b<&JiA+sja$2Z!Q|=F_oZxUuN!Sm zeCEzsN#oYW)3^s74-K_!lUBCTyx zvrMY&IGlo;6e(4zo^kcqNZF%nzovh)s}fAo^VBI<<#p!m07ygKfA7yvk+{?TZ1BFaTLx!gpQ3uGdh zNFaI=h!iHt19828kSOPgM1)9`Jog~I6UfR*PD?2QGLb|g5Ge$bJCj64$nFS<{5J4w x3`+$7R)!4`IVmVVF_#N`KHnuRJ0p*i$m6=?+nku{R{Os%dP+b literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_copy.png b/Media/Themes/Umami/Icon/actions/stock_copy.png new file mode 100644 index 0000000000000000000000000000000000000000..94c72fec54fb6c3bb50fd7a13d797816e8ab31a2 GIT binary patch literal 891 zcmV->1BCpEP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_first.png b/Media/Themes/Umami/Icon/actions/stock_first.png new file mode 100644 index 0000000000000000000000000000000000000000..1822571ee05a1649593d5f07a36501ff2040600e GIT binary patch literal 1140 zcmV-)1dIELP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00U7;L_t(Y$L*9|h*eb- z$AA0doO@?-s0L*wsf!k&Yea>dB|9`Eu|9iv# zR>KrPp~s#^XvysUHqe_ank}hNviB;@(_g)wtg1y|0T5cA-28m?Q-rto?N~SqutKrp z>IH4j{Ql?94fS-Ovp*kxzjM~WLdiBM&4HVjv~PR-?(KDf&va133MIQ(Y5LdPy5Xu7 zOIHTg1oh(ACMcBbI%{+LHm-i8X+d+VIz03lx3v{fhH{}~<)Q?|@IY_US{Ff~$8Nu3 z-uxFgt=ZBzDPtTS+K<&C-Y2~D=|!5k?0H9*+m>h}Aiqi%60(iH}gC zu_{k2ll336mxS0?h+P?_Du{S=C_sV@G}+C|Lrsv=gn3QasL3^G!UjW-H)yMB6HpbN9Qom!FZu`ePqi<) zhwGYeLK}s)3KJ+ys0i{^;IPpU7Ixl8{9T=cJEVnuH2ZPe|ZZ( zjvquTMfdeR7^~2MMr&}1;9|ii=dVQwlmVIR9)w4ByFE8_|26ng`Tf_{b**pgxTdI% zj`RW0`|it;izK{+3eTj4TOgi`Km+DvpZ|7ZXlhFwfBaVaEnUqW%kH<{2`-Xw;Ekjc zsH9%bta@7Ta}jvJ0q*oM`E}xt%I65;mIx#jY zFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vgD=;uR zFfiv?T%-U102y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Ny!L_t(Y$JLZgNK{c2 z$A9O{luTMW10mr@21*D*s~{}WjUau2MQx%MEex`-7D*5-S{UODEejF~S_Fk)i$2Yw zXj9Qvl0Xsdi!$>gC1)JxwRrcvc{9%B1h(kFedpeH@B9DHIro1a{O=~tBpm6y2DGja zZuhy4^T*TPgd?5SN?AD4X*sz5OzN$zx6;;6x|*}(EeHZwG<7^m+j`r5IBl)Da#R%; zT+BldL~Ljg5D}h6ccCie#K{G|Hn55_S6?HdAc~-%hCtLEN5d~zQ{cz%ll%xopfB9a zhERa&vSRMN8nvTUP;m(CjBY<4qOK}LX)r*4bPv8PnJgjbkL*T7xzfIaiuFa51_QkR zFk?rI*!34-=+g|O3fZgXyBOdTDku7nT^EW;PD~IV$7h+Ii_`mZI8`#qL#7RG(MbkIeo#{pbOD@b zD@%(R7@6SB;8*rvcnHAulRe85SQa$2_cHl2#>kf$P950dmd%l{(+fwMY8e=r;QRD1 z>f4rrXC(kgS`b@U5xAmY-bD^mwG(?E9?Zlm2 zpZg$-Of!GC%h?~!OBz8Uk+{(ny>#ra&~sQk7QdOs;z=n06dLb0CG>md&HzZ7u>d5D zj|20-91sJNnZoq~iU&|U-sd5(fJKi_8oyxX|K5HBKC0%a9W)b90000rbVXQnLvL+u zWo~o;C}SxgGB7YSAWC6tbz&e>bY*F7WpWA%Lvm$dbY)~9cWHEJAXI2&AV*0}P>|V+ zZvX%QC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK z03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$50000< KMNUMnLSTXk_mFk~ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_help-add-bookmark.png b/Media/Themes/Umami/Icon/actions/stock_help-add-bookmark.png new file mode 100644 index 0000000000000000000000000000000000000000..e2e8d311022d678d82efff324ceb848479779b1c GIT binary patch literal 1101 zcmV-T1hV^yP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00U1+L_t(Y$GwxyZyQw< z#eeV3jK|YyaG=44Mr08w3BitHDFA=3K11WAwXH8A|bKkFNgrK zKwYBH;z|&rTBS~twEn1*iK$}8Ut`Z>;SZ&*-8LXN(&*{EdGDTk?ztEKV~AddbFZBH z7&zbekr27dKYscYKt1=G0Xzr#9C*9ntpdAxXvMpa;Ij*#ECLGv*Ahvt=Rz!;dk3Z3 zft%~%^0!2Wvy4xSb0SdYrP;d-KXV#sj(^seLwHpZgRS zCICu-rW;62XLNLwXfz5y#cSYJa||7;6JIx(7-{_0Ho3n7O``iObkoMPsx6}xhTlRr zEH*Ya7#th~AQp?EJ?1WZ2H=c+=cD!ZQ}7b@pv4=Fp#omsNM=*c>WaWbQ;kblYwI$k z>?kK1jWRPcgJ^YEdwnDVAq2%@5dgCOQV6cEC)vq5TwFTKPS)W@;wAv$aF|FW!u0esQp&yxdeZ`azaLL2cCrp%E}mg9 zFu=v7v!u2%NGY&w8(r5AHN+7-_~r5{<6{$ibKz}LsT8Gy0_9Q>&+|C{-pf?0Rg%dh z`FtK|wrM}~BshKQMFN2Ug+hT~FxZqTWv3L4Mg!M%v27awP1Eo^@8~8=DM=&}JPff| z>}VxAjLFGK48yoDfp*=SQ-4GS?I6Gd75w8lIE-*O+{w0`ardf}LiL&Wo&=##=%ICg zRNO=0dEV0JUo3z0Zvx)Fm3sH^Ui7)A1IK}Z)^{ARX%40aD8L2Ez+b>VaM0Q}TL1qa z{0-kuCedog!JGg903~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eN zIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)=GJFgh?W T*0Dx800000NkvXXu0mjfL&DnB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_home.png b/Media/Themes/Umami/Icon/actions/stock_home.png new file mode 100644 index 0000000000000000000000000000000000000000..1d654650541eace316d850109a8f8ec79e084a20 GIT binary patch literal 1072 zcmV-01kd}4P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_mail-reply-to-all.png b/Media/Themes/Umami/Icon/actions/stock_mail-reply-to-all.png new file mode 100644 index 0000000000000000000000000000000000000000..a0d26a418198f0e03e3e9b1d1cff61d92a989805 GIT binary patch literal 1409 zcmV-{1%CR8P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00e_cL_t(Y$HkRtOjK7G zhDk$hFlpis6XOpPEir90DhQ~Fh*&EqE_7rsn@mAw*!QIf3@UqBgtRh4r3kbX2805 zl49{s!{xYLRVyGr{u@O&Y6aI~dFbx$#>&bH6bc2FmX@%%xVV;ug@p~hGV}BE%E(9!ci-T;ziIqAVWl;1kBCNQOJ6r@;W;^OM8W41wvsFT~9b_1>q5)bbu_7NF-E3 z-~_Tha=CmhWH$=MGQ{F?x&u;#qgD_c8b}A^@|RGVOorvOtj|XbIu-o#?nNZr51!fIvBmfqrl0@Nk%@6c?L{+0&i()qF`nI zQjpF$h5ke%JmH(6=PPGCxWz(C+9?#c?0|frlLFVL;oDjaJs(gJ`1pCfLg2UygSi%H z6A^(nt@%kW)6`#lC{a`##JBCN$x@hU_#ZI$$ z9I#GZ6GUcKpzDVb)VB1hO4zJzXihXl8BY(zq3@vA^$jew^Dxm6gee&xW2KfD$undg zx&umVw^Ml6lb`67{C9}p4402O$pHL66>SulSQJ(*jMz6}9&j6B+08g? zle|HIkO%|#gzi9G+6fvD?p_2KvXE|jf;kpz3<{l4o$XKI)skn(5cr};;4(;8qO zUyF;pN}P=l!8EK4CSm0`9V&*^^;+0OS3ui@_f)Nbguqh@B{<|!f+OA{lvhjPaz2U* z1Y#M?BdaLzOjyMq2^GQO6X9|ozmv^cA4l8^C^#acPAT~NWu@z!L?o}v$zy>OzSH#jQ;q+tegjJX zuxNCqPCozu03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn z^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)}MKFgh?W=UH5& P00000NkvXXu0mjf>E2o( literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_mail-reply.png b/Media/Themes/Umami/Icon/actions/stock_mail-reply.png new file mode 100644 index 0000000000000000000000000000000000000000..e769da498258b2f43528fcb2c0aa1c6212963bb1 GIT binary patch literal 1181 zcmV;O1Y-M%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00W;%L_t(Y$L*9|OdM4h zhM(D)0bJMuEv1kxG=i3t0LHaxQoSK+sxe-ew9=Yb(^#Wk8IvYP+DIB2Ow*)Itc?bR=v#N$hZLLn^6O1|@~6bJ+Y1cN~U9LGUQi4X$Eagwmq zz7PUMQBYOr=(s>25V(}Oy&%-rHKy_f$?<75aVpa}~s%fc`Wiv1;Y|8R*9&UNtcbMNu#sr}3? znB2CticKXY^bJiAoin-68%pD>X_1r?DJ2u(NqjyZw{G5owyu;T?+nuR^5evezr8N= z7cn&S8JWa@XC<}ehN)fv!#HIg}L1R z$Qk^_`4kp-841rZI(99MlOd2e-?nYzFZC0dj&Nl>%7Nzdyw20X zxpqB<8F#RNnR$~O)lTEADNtHkn#^}U)yAn8cXPgHg!#CQVOp4$z!nY?1&{RL^JU|) zmeI0vZ!B9*?}s8qk=X@wU7@Oc1Kw;6pJwy+$Cof|d3LnD z@!(2`vI;aC$V$n2@m){UdpB)6`|M-&GgrdK2A?;J_OH817~#D|ZM*!@q2~b!ECX@3 zpCORrf-`c_9gptF^9RlxI#54LPJZFZ*3TJjZ`=WxE-*DOxk4HDAhpN@47WHI8T_Pk zY3kP}PriGxFg!Iq>DI@9MYk^Q)-ELAH`@cn{Qxz!uKa=Kr#-5ETH5yTSjS7XfCC6O zuey)w_r>w{v|EB3xV7sDe@4I8xTJZlQ{sLuYFFqT@1a1HT03~!qSaf7zbY(hY za%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0 vW_bWIFflPLFg7hQH&ih)Ix;spF)}MKFgh?W=UH5&00000NkvXXu0mjfkUJEY literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_mail-send-receive.png b/Media/Themes/Umami/Icon/actions/stock_mail-send-receive.png new file mode 100644 index 0000000000000000000000000000000000000000..1bd0d8df90c56714632db94b37d690f3b5d69e04 GIT binary patch literal 1111 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;JpsQj7}P}D;O9UcLw-`xH8ZTD0sEYxi^9w>eVi060hLZE@_=3VVNPQ?jfM&CTb8V zq93l{)h_SRqTtn@nUSL4)vn;);+4OmVg9l1Wv5e`*H2q}I;4Ef%#9b*TDQ#Fd_JRd zvjSMLqI0{VPlu9Yhmv!vvQvk$bGNczm#SZvntPvl^emUmWo{K4+zZ#Zmu+;fSnpZB z(X(Q+SMf^!+O09BX_@_dbF3iYsRwt*$DqsVb{!>aJaI zvZ1cBVbSr%`l{x}nwCY!+FI({TN^ssn>v=A?&)gn@9pUC?VPas?378p3$|Zfv}o?q z#S2!gShnuat&KeEjIq)0gj^z54LNjmMWSoVt4X z^3^Msuikla>-xo8H?QBnclYU&C$C>T`taf1r%#_gfByXC)2Adu|NsAuLTA$$7#OTdg8YIR7@3$^SlQS)IJvm_1q4Mz z#l$5frDPSARn)X~^bL%Rja^(qBcfu`Gc&Vm>!;s*`0&N6SFc~c|MD&TtGyBf0|R4{ zx4X-vX)Ny*7#J9XJzX3_G|nd{EHDe$wyp1+hqtF!M=uvQFCW*H*6!#h(bs*MnmYgf zt=%hJwK?HYS{>ujYoWGxZEf$~jlDa$Ytw>9Zgvg<39&bC-n@D7rtQilnzAbEQga_Y z=A5Ltd0j?!ZvO0$8#iY2h=gx3>*?(L_>ob_FroUBRCLwjMj@+T*RGv2w2L$J^SgHK z-@l!fybhfnD^|}*+H?2uoI4E%_V3)aXWLS98P`LH4<5U|f9KA9Teg|A1TPZj$jaK_ z(bB+p=DDAs1H(iIUM=-#?FE-@WZv5TX>{QCbh~)}z~RG(FA928{5Y+6L{f6=-(K0b zGK~ti5@$GBB9-*}@P-LvDUbW?ChKk%6J1u7R1Zp?Qd*k(H6Tm7x(t!}G|PC7}H5>FVdQ I&MBb@0E2GFTmS$7 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_media-fwd.png b/Media/Themes/Umami/Icon/actions/stock_media-fwd.png new file mode 100644 index 0000000000000000000000000000000000000000..6212e20ed5c18931b307941f1438e05df798a6f8 GIT binary patch literal 1229 zcmV;;1Ty=HP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00XK?L_t(Y$L*8dPg`{u z$G^YdIj22+VTHmd(pSu)61Vm3fK)>?v2A9Qwlbrc+q_v~vI~NVE?acb5Uvae#p#V% z+%9~{BF+oL)GcIA{{)5!Dlb7OEoCJwU3>a+&d&=AW+1^`nV3B{zvtrnJfH9LOTNH^ z`rj8I~?xL*&&{FG5_Y1JT{K`vXEiIPj`T2!~+5&Z(w(026*y{%#InbNS z=4SKx{N0+Ne+6hf>^bz2Px5tJEc+vRqhTqrlvo2G17MpMJQ3&|^msiXQ769kP^+af zH#@&j(3gm&;}1@w$(N zf?*3U@aXP-TIlTPG%qbJzF@UmeF{PRjPFPx-6ZIyO6bgndRkZ@SSb)K3Bt6eZg|;@AE4h+5Zg<-1E9L4O$8mD8 zSS)V=y1Ked=EjEpXf%dOr3{Yapr{HIr3$axCj^5bLpr_sOfH)>heBZu2vDt7Q7RRY zzng=@?xeA3ub|iK4sYbMC#`nts?+XToSmI5Pymo5NdW)?0+p#y6a|V>LAhK)K`y}D zV8*M*j_W~)o)8K`C@7^6G#UsR4N7Gh9OV!W9uXZ5m(gG_JSRyK1ppoZ{C>YOdGp3! zf5!3PC#W*7vdF4 z>4*J&Z^Sz~I@SOfK?nc_ov1@5lVKzA5qTw<_|fC{zSDoY{})2Yj)e1D9Rrf7B#XyK z#nsnry&e)DzV)=#&Vtv35xmo8qO1W+J^?8v}90J$WW z&z?K;yVKS7=BF1g{rpcus>-s1g9ABJy?Nlm*vFSmCR3IWvTKU}HqY4p@$vD;*4Nh? z7-QTnntfAKQ}&682`6Jrv**%n!#rb5*h51ZV_L=-xv%99>OZeP0p5CR)!r$^&Hw-a zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuG rZ)S9NVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjfT^Sc< literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_media-next.png b/Media/Themes/Umami/Icon/actions/stock_media-next.png new file mode 100644 index 0000000000000000000000000000000000000000..56e943d7d5cfd75cff894c5d7f2090f1fe7ea78c GIT binary patch literal 1318 zcmY+7dpr{e9Ke4=w!Ar?i*#|ushGA|nng7x^YJLJ@~W`LX_&M&L}qDBrSnXv594ml z=t^F-JW`ELlDQr#EV-0d;XV#GcAxJ4y3hCX{qy_z3J&;s=xS}$0svi4D#afFOumZJ z1ONrcWeET@V#q#ZfLoWem67WJz=B5g_W?-#0f3tYumS+wj{v8z0Fw~_q*8zli4}MK zoB=>H%FBZSs;^YGi3$N=eIhmFBmhwVb>CygZU7B+PYOAZ+C38(=Dmq-;xZb}9=dv% z&A0CCdTL*K#Q(C9v8yCL6jRjrBI%RcdpC)UP%NjI*J(=0jXt6E2Wy@g?A=B<(g2wI}NFJwB_$Tgd9PESwApD|xFx?b9DlKv^BY&>n@>StAZq59iymG0Wn zSr2F}*3g0;5W<+PvPLZta_T!p?;3~Vj(8Sm={)%98v(_5ZxIr*(b5`r_hO4y_;R!O z@zfx&U|%ZA-S79Z*ONPqWAuJudiM`MQisoMY5Mb~OBvGWzR3rSbcWflt7$WME+FCQ z+iyO#mPfI&AuW2u9U=)}|u%&>Hd-6W|EZ7)K0?%GrkIKIu5 z$u!rO&CB0Bp7zQwrb;?6s90cKFt6js?opp)PE~ zb}~a%(RQoq9o3Dtx-6H=FM<{VbQG<0Ore_4!YV~p>A`w3e5bFte*3~T^VWzr9n8@+ zc>@w@YN|%lO3$2E-lAUEybHVdtzBaw6mb2R`3ift_GuE>jqhC0z)G_SKf0)M3{_H~!IWt0$|FkCR= zbgJxjuyxKo)NLp2yEl-bM;FWrUy9moS$9?@Ht@^^Jd|bV8c(TAJG&(%<@S&KV@+o# z&OK8okQtlfNM8o~${n(ad0|$T!5E{420Oukq`G4dM{#9WDk1gb!@0uk$!*ubkgRw_ zw>2u*efgqTT%(?RBqwcU$i@MUzD(fc|NZLAIj=)}KL70=<&q0Tu-S|JQC02P%=CfD zjMc{eiE^YtbFf_kC|buWYil=}A|M!LpL3{V^J=xIrF)F^q5VyI^Muf@cR{?Ah*U$I zj5*XjzO=LyKD06~3JVQAgD|iXqG)!5-Pl+7axGBI8E5nE_h*>36P17TZf}zZ`1v_$ z??qc`0*yxd9#iMztt8EzILcIUtvtz4MbPg|yyHy9D0R!q^t+;(d_~n+{%ieS=~H5H z$%iNYyEP*+bJo61zJN0WMs42DlYbJnVr36AsGeBalM6WQS%l04Cu=?Ze7HN~dsTay{{<5v;WYqY`t=aNjAn5nljy+VaO`6f7$<3w YiFA8rQcU@*^F{ywPj_F+4c7?XKM-$2EdT%j literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_media-pause.png b/Media/Themes/Umami/Icon/actions/stock_media-pause.png new file mode 100644 index 0000000000000000000000000000000000000000..378ab5edb36c8fb6d2bcd0334c988c8d96239c6c GIT binary patch literal 798 zcmV+(1L6FMP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JOML_t(Y$L&UPt#SW>YS>YZs5lLd!wy$A3$7#{Q=+{z!*TQ2J;=j*RsPYAA}IW;nDq5DdnBA zUrPDs^!qW$GN!4U-`>5Q-DBo@0MuS^L z{xt%zMZk5-YHH2yCV>?F_MG+Uuq;w#pw4L}tzyJ`(vF%{FTH^QOCuqY^j_o99x7!wQ zd2s<_mcWLfAWxFSs&u48L%BA;9IGR+03isDqpZ3pjPkMwBA}rfP)gYmb-UgD1pR)W zi`^;JfO1sP0NU-fy1`&z$EZ{Tj?=IrdcEHM5eerWTo!sekc!Md8NWm#C zBoRiV(f+?^JRVy>93>TCx7*DDU^1Cll}XY@UH(pj>2#V4A?8n>J&MYHrBr_V;NTJf z(lq5f&wo9B`mpjA#c{Gq)AZViDmR6hTUnNQ#+ZuVbzSy6&k)gqnOnvfFUztD#`8RO zU3W!9OWfE$Wq$!yIhrA0r1(An001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHY< zIx{soF*YkOFgh?Wmh(6<0000bbVXQnWMOn=I&E)cX=Zr004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00Ux4L_t(Y$JNx^YaCS^ z2k`Ij%$c3NZ?>h$?!>0aY<8Piuh_ySDL$pF+tecDLTcMTpvH$HtuMs~LGTu^Ao^l8 z)h2>}MB)t*vHI2&Vq4OUW=$_VEf*E&wUkDBNK_;iQaOlay6IBRRQP# z>_-AT@zkCZpPrjh0s&tlm(LyT+O?CeFK058i@=k6PE1Z5;hGj>=@-(jLaDGf z6bwwZZ4a$xmREB1mWeF^W258z*6kbMoMUKUKOY<#@Ml(54s`G8dR}5!T3fqm0Wj6jWD9p}2x zAu$FfF)+qJ1em6QYPE{~zQ?7J^g*RiEG9iZWwd=q$2HD*K~+^pRaGOK&29`NlgT!h zBELE|I?jb5z?cNaSR;uDRogM>G<}$jBkLX&HmTpf{pvTBA+;ULY2W z**9+0K97Nc7y}~$5dl<}9S5@Pg2(N}YIYT8rq5Jzchb;jBNfKl(0|4e1=8Z2dd|vVS{BOQ?^7O3N>&?gG@eK!K6VTn=-2iOc1`$D) zCAef4mVR5Zr>EZ&k|bR?_VV#}QmND$5tY#*ngB&n8f(TFd|p3t`8>{^{h*S~{_*p( z&piG5)YQq#0E$Gk>G0#NHWil#vvaeRg|8QGMmsuAo%>{F9zcPJ%*_gK0yy~Q+XZXp z{CVB)_r3l8yVIYwx3}kssPdqNTLOyWzBKpc+~VZKk+;(6^q&vfj*Wtv-raF_pmqUG z0C$MUdYHoZ4aad@j^ij>uk;_q-%^7?TS@`r?f?J)C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML zHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* XE^l&Yo9;Xs00000NkvXXu0mjfMiBdA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_media-prev.png b/Media/Themes/Umami/Icon/actions/stock_media-prev.png new file mode 100644 index 0000000000000000000000000000000000000000..070ce76f4b8a0ec02ff44a520bcfffe63f4eb3d8 GIT binary patch literal 1246 zcmV<41R?v0P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00X;8L_t(Y$L&*1Ok7nI z{_dOi?)#ZxW|)zIN<_w?I)Gg?GXb?Jg|--qt);2s6qFz9UkK1itU{y)B3dOH7lwqA zrpARU6Jx8i$_h;kv5UGWi5sISCWN%1fmayld+=tCibIx~w z`}n`%u$>tI;olPQDvDC-bZSM6v3$$2EC9yeym1032A~00z6-Ab*am+US3}5NhX&VW6Yi-BjELV2>>LC0{|W@E>>#x>`s(E5`6Ad|EZKO z-|y3JEg+pv!{y8a0PNjg`+Zw$+j>Joqh^|hdtqUI)4*g|hH07^fr`q?6k`mMD1kA? z9^X-Mp{J+Eda`<#`?j&lmX>q~f&f9tm@X2Dpu6jYM-&~dm6c@>5d?<=>2!KSAR3LP zrlzJczAq1lQ}zlZF1d<|wh6guX4*tW5%uNFJEUg+dkph}j|6*VoT|`2KLmjq7v$?H#SlSAMzjyQ*ku&6eQWA2Y8Xf2IAm z^B2CJGEIZx%`t0w`f3RgiA3a`oSZCcf9b`~ySh6U z7WKtBBBEIQC=pRW*L7ba(fw+?Id<)bA1D8`ENjiq-(gdG4*{eYV>%HTb#-;U0|NtJ zb#}h?NyCB0B!KN%_bm|77c(<67l%Lk;P|=E&Lt9wL=r%7lTCd-za@&I-qz9T=bY<4 zpWoWpHpUDhGKYqSewv+~-T&^;V3#O~dvcqc<*$#8p8a%UV&coOu`@lQC{@|F6-1-~ zaAjixkYtRlVhe~!0^rVmEX5eRxrKKC$O9k)NHfOf?Ia^2fruFH0U}a~i2nmG5s`SW zJooY6;4jwVU^;i$+rt0=03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`If zHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)}MK zFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@WXPfRk8UO$Q07*qo IM6N<$f~q?q`Tzg` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_media-rec.png b/Media/Themes/Umami/Icon/actions/stock_media-rec.png new file mode 100644 index 0000000000000000000000000000000000000000..d13d3ef8b8002b4faa73d1b26db642568e6aefe2 GIT binary patch literal 1258 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YNKL_t(Y$L*C*Xd6cq z#=kc!X=SZewpVr}Ir2Znq$w?N6(u+n(-dNygJ^pQJ>*!DQ%Kt!n$T0wA%@<1XbJ7L zp{3AEO9(l{g%Sv{n^47R>_do+k!nkdEIIXRrPZ!>rUzTZ|CG8t=YtP+hT-?U$2?|% zSNa#x9?nPr0sy=Kyec5;KplX&;WYrx{|)e~K+m_phY_L^K44f9qT3;QRs&}K65_Y> zz)x#H9zf+k1t^LV`V=_)UbXslcOuc>l}yT>zCO6Ry3i<`QgCDi{b7wXIz|f*yS?J#YH&PDwy97ZEVcde&|qF z^xnPG2N?U~Ni@2=y1H6_X+R>8a32O@r*pX<4_>*_>FMt$4>Bg-fcc1~>C|1Ox&BuVUNp2gk=<)rSv(YIRq*sMi6@ z0%|qrlamMz54-)A^+7ls_O=X2rBbeUGnt6HypEcD;JF;Y7zAS=#y~{t76Obh z2q7QQxptij1 zSxF{m0a&f>fYa#c-$ICjT_`|~#Xv;Bwvf$m5J8T`U>6F2BrPi=BYG;8YCWc!rd6C+ zY+~u=O;brEz!YU`7aMM_B}GBokt0~Tb<4B{2fxrXZOfCf%ipS$QA`-2p2t*`=5XHI6 zmp^+pG4XvZpC9Z=CcV5`1uvB#DittQ1q%j2J|A*7ZkT!|v!GnPdg--zd}T9y*AAM> zX0vTwlGOd=`0)#(Q21CK8WKt*B6~EAS>-aX=5qD5r%#DOp)U_kO?|WCIK|%H-rA0W z@e&CkBmkYoYuBQs>(}41GMP9kl{c8GK6dr?rnR$Y)BDbyTLw@jBEB1AFF%}SkPN`1 z>$*!-)yBps)``g33*(hq>L1=SsHBN#{u}@R03~!qSaf7zbY(hYa%Ew3WdJfTGB7PL zIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQ zH&ih)Ix;spF)}MKFgh?W=UH5&0000PbVXQnQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@W UXPfRk8UO$Q07*qoM6N<$f&eigfB*mh literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_media-rew.png b/Media/Themes/Umami/Icon/actions/stock_media-rew.png new file mode 100644 index 0000000000000000000000000000000000000000..134fc02c9598f273f7f29b75789065e12647ea05 GIT binary patch literal 1184 zcmV;R1Yi4!P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00VtVL_t(Y$L*9|Oj~sr z#-IN=EvM%cR@yTtG%dPe1hx)8T5yDnY?MJrh-H4^2o3_BrQL?*LSr3Flg*i!ZU)UF zH{KXSyx@*3yI|^#=$+n-5qdf5OelkHjDDVTUN0!5elX^Z7iRC>Kl$Z(p1gV9Kk#J! z_rw27z7&8JfC?bH10w>c03ZNJa*ru&wA<~@m%F>Jo;cxs3qZ*~FnRzs&vDPW*Lz-_ z$cKS$7h%C*Fq9uT(%dhXS>CO$e_o147k>rtRRO~RurxI_9;Y~$m$T2^E+W_GG zM;Y({_BJ;k_Sm0tTx@J;tQZ>fNy%gqADjj|4Rnt2oqoS?S#**ocLas=btAX( ztbdqX2S9%k+%~XSEG4omJL(SBo|P@;fk5y~(F@PL$ge-VkJZEq7~}E@7(DGBZ<#EQ zd%a#`clRrzqGYj}h=T~AV$!UA^-qs1jcc7t*ysC3HSq&BuPjnH$X(-I3Aj&ZSDYo+wE58=H|fVADm>Ay7pIB zOPvjfj#B9q z#aH5V=3MaSXf$dK1pI-%Q@tzKZ(L6)ibATYQYw`sP1C5azgJmUSg6>s{6u7kOhjA* zgZ($hKOEb*w|b8>O`~{xna+*`e124tF^)t8W|M|o62^% zWDpS_i^WPWU%oVyc(68ZGMj(=@{7qg8DsYg7zV&NHWog5Cw3>ID5_)Xn{W3q#+G-} zFEkQnW@h#;EiKuJXwObz*4Ea{)6>(nL{#?J;)O|xNXRFC6sISm-Tt{J^B>J0j<;SQ znoAG}0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ; zF#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vgD=;uRFfiv?T%-U1 y02y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s79PZ!4!kK=EH?|Lx>3j9kuXZbp9*P`xv zm5CR+OT=Bu1uwpK{lX=>*Jq|nMY_j=w?5*lS3g_EuDGi(udJ->+i&aL2iE=DopCyS z|I}YiCX6B56eJj5==0^=3zKN8PJh1q`{Gp5_i$uwjhgl}`@dwT$E@dG3tm3hP|U)Ru{G+}>#t_QCQ`g76*?TA zl;10qIelL8x$d>srFVI@-}@QG9w_qic>;^l0SWGaO?wvX`e1u*w@u=Hp*{Tf7?gU{ zlWSDt14LK6`qH&VeCG0?wPAmiUX|@;Vqjr1K5TG!L#wISHH(PshijTQGctuo2t9kM z(D%4xV#5Jej~>RfRSUWOF4ZY2eN^H&9FWe}w4g!7)5u|FW4GV?=QHlL+jMg<%n1Iv zXv(V4My>^Onl_&aepg~u$LF;)Xqk$Z`-inj8(%zM5TmCpUAJ@BKAUf>>5t+z-;61o z@Vk4}E4zk6Zdb0(61aF=?ETwU{I8{p{x4FvmJ_GJz`&qd;u=wsl30>zm0XmXSdz+M zWME{VYhbBsU>0I%VP$M;WoV{tU|?lnFzvI2A&Q3F{FKbJN(LhXLqlByGhIXT5JMv? zBXcW5BZ!9Qkuggc7#Ji$HU#IVm6RtIr81P4m+NKbWfvzW7NqLs7p2dBXCuYHz`)?? L>gTe~DWM4fwTTSX literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_new-address-book.png b/Media/Themes/Umami/Icon/actions/stock_new-address-book.png new file mode 100644 index 0000000000000000000000000000000000000000..1cdd0b6b73d4d00e6e426964218b6554a4b2f2b5 GIT binary patch literal 1083 zcmV-B1jPG^P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0Q>;y08cd6u?+wK010qNS#tmY4c7nw4c7reD4Tcy00TWqL_t(Y$F-HsZyQAz z#(%RP_1F3{P7R5PNSjt9ky@lmg#-TqpdJuX4i(}A+~H3^NL=^>;M7YMju91t1K*X1 zK&lYJjpZhkki>D~xQ^@fuH)U=<=_wO#I9Wl9%(eAcV?bv=6&9sh5s1>z&*I314}>V zE_`tB&u4CKmZqp%0sQv##E{Zm{ruy%-vV5~2hcimmo^XP?_68Ho_*!zi#)0p*gSfH z<7R@I>9kKGx+fC!>vo5YW{W~@ZKd)DE3V|z%vIj)7g$Ip7qiCHt000T6AI7EJilx& zb8Jp_`b;VUKz#ZJwes3KFd(4U9arLeHbPNwPWExVU?)XB0E**c6w3r_4TSV?!a?e@ zX&fW`L}8`fIqP4pTiCU4IC=aPM@Ji2GK=d4+G%hl$<1AyNQm;%Jcm~nu+wppTYKoH zJzhY|tl2;1&a-DOvtNr+C@o^k=X&}^se+Y`QsbK02RbQ7`5f#=_O5jLAVlMRZ&kI0<{Lh z_X#{UX+>Gq)6wt@fCsaJRe=bZJm_){;m3S-iqwN0>a)`v zUb%>`s+copi7$K|<@G#V-LUv%n80v)aKk~W3t5c4GO?Y*b}XP%S(;}Yv|+(e@Z6c0 z6A4khJU5nbtOo*!MnX?bG}#S&&~=SO!l3E6WRekLp#Y;ERdIT}m1ApT0}DPJDquEj zzPNK2P4V&kc5_`Hp$LSa(*!~Yluj=IDLww$E&Hw4-V?Wfzk40HKUCnmZ-3>b#R!Tr z`F}uB6mI=chETk_$)m4Y1_W$ZV#Ly9XEGxRg*Yqz;)6QB6?xnDOXp% zUpKu=^OmA(eg2Q}U~VQssd(QnkrK2EM-1Dlz4`uUt9l^x#>Dv%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vfD=;uRFfi7!MmYcg002ovPDHLkV1j)9 B)t3MO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_new-appointment.png b/Media/Themes/Umami/Icon/actions/stock_new-appointment.png new file mode 100644 index 0000000000000000000000000000000000000000..1c55f86f3fb5bdd09fc211bf84a9623574f81cfb GIT binary patch literal 1477 zcmV;$1v>hPP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00hTLL_t(Y$Hi4^Op{j_ z7GgHd+|*r?Z7xHz#o=NWcVSC*iCZ$Ai)4WUD-IW&Y*{uWOlZ-x;8+%&OZ@K^ll@VY zK?Fs)h*%&Xf?K&(N@*#*(F-lTmx5xSz3-8dFxbqpAAZS`@5?#g`#kUaocElwV#R}g zWNC`Lq70=*maYuR)08p7-Qsj*uqa*8MRWWA7crAoo~VMl{!ff_Xfb+yM1a3}jOU8e z6rSVX6i5-;xE+yApJ4j=cYv2y{VREqM0x(?;ZxA42BE6EfpcdrqpIpMnwnI&bg3C< zD{ArEuT^NMyoSz8?XlX^U%Gs@Y*# zJsW}3Pyw6n9PIik%*;gL@wm~}+KGk>hhfm{L|`%_HXHiRkLV5u3Dr?G+m~jZt|;Kl znY^~90ijTU0{_6cx*Vqd7V_v zNRl3*q`V~p@{H0b+fq~Ah)CE68iLb!9_IdIBvOjet^*iWNf8YCiF}Hxa3T_{z@1Po zqQRdq8@Ym@OF2U}4UtW|7m*icDNeFAs=DhK8#5vluw!~UEJQ2=$6)9_0A2fDXxly) zmHGW%Oij7r^VA?3`VqH7dGI^3VY#&{*3{l%+-AQA~7;G2MM zxLb&fwC6x`HI)Q%770vF+TnGdMR2MdR>M(rT|SMHJ#zgLdk$qCxx?+(($Xd@V6aGZ z+7F{%Ekp+D-^Eb#?gaslXA&-#19rO&Hrpurd)3giY9Ny+XBLs~lNPU(rz^$c>PPjs zH9R1)Vu#IGuK~bd+KSF|4__Ph_+q|1PjDE?}{k(cDrG zi`68$g+;>QAX(uQCxYA7|lY)T3ix?;8ZysY?ICCa9 zo5x`^8feamwu>F&e9D)UR4x(tVvqbKd1l#N-e?28LxPM+ILpSydhurP7+1u$#L4}B zABOsd#n~mxD4iqA*DiYwz{VKKa zG*Kli$Dgoz)gv1UKUI3;huye=J1H>lwFPY**P!mZNupL^RUqJp?v`Hs%Q4@k$sEM9 zmGH!J0uK{5Ccn5XvrwuG$B%1WuMU=R3+;<3czs@xvCrp&c2Ea(YoGXs|IL@k{p;60 znN1vT5T0F5Ad#?v@H$nJwl6hbPiqpp--Sb-$Fe`%Kb)ADm`kxc39k^={L@|Y5LZEX zlaRVKIpyg7v_iW)=ZCwp%rf4g_mEMFB5B!NZfd?`{o1FL^luK~Bf_hM$L@Dq@&c_V zY$3c&*!Ad|b$iLW@85gt<1;DS-v1@(xy{OigoM2`Cn4-0q!6AVto*N6Ggrhd_w;;a zTjue?yq%BFuib-t{sJ4mQKv9DxJm#303~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_Y zR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih) fIx;spF)=GJFgh?W*0Dx800000NkvXXu0mjfM8Bv> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_new-bcard.png b/Media/Themes/Umami/Icon/actions/stock_new-bcard.png new file mode 100644 index 0000000000000000000000000000000000000000..9123074d35799adf58778e153d4cb96140c7b429 GIT binary patch literal 1032 zcmV+j1o!)iP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Rn1L_t(Y$L*C*Y!p=- z$3Jgoy4`M-&X$xSP5+pPB%*cs)1pM}Mu{Z;i_`<$3P-%^)euco)PvD@LFAt_1`-Pw z;|XkH;>AM~s-=rdH`_Kyi_6kthwjYmdp{3_q1!GT8V;KHOVX8$J^D1rW-Li!RFN9%S5=$4_&@+M*O5Z88{tKebtrWO&C=f9 zPHSr`0|Nu3dzx_{+Q_Q<->~#Ah$PWgn)5%0JE{T-S4Oeww=gsQ7x9J!Gt)VYF%${~ zGMNmCL;?{3!1FvtE*fs!XyB!umuYOY3CruTV=n+-P!*7Dc^tnuOS~b0gaPqYYca;q z-Q7)BR~I6JF$NK#udna+{5eDf&tLEpvAcN{fd13(GBh9KYX1*7fni=*{OLZ%$y4vJ z=dFX}a=GZ~jWKs$RS{yb7)?!0(eXA{se=AN7>!`F>P&a z6`^-jE|;TKEe0pHcY})iD#u$plB0&&D;Ua=kiexfb(YkvAdS8E^ zU9TNr&MQP-ePT71Wg{ZEu8W9Nge{&|15|tmlarIg<8>6T&y#f5alLShe104eVQg#+ zr4;#m9;FnvZ8JPF1i;AX1+>C3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgF)J`IIxsNSu|_!n0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00Q_)L_t(Y$L*BOYgAVl zho5uqnV6Y6@vCNt1uH}l;$+cAkP6cN2ku1Z%C@eGSl5E!KVVi;s&v;)5EpTyD;2U3 zS~M{+VJ7hd(IzvQWHQNo-MJs9`{e-<(K~{&2dI1A2f-U{Ynq8vrUEU})$lQaW7$ z2M&C;sqHw1{ynQC^u$HaIaWgSDW&tTgf5H*#ZzM{Gv4X|JkMipZkBuVhN(ZdQ(eR1 z%sStm%=6`uF9_6`{f^p8y*aUGxp@2XV5aa0nBAlxn;qur{b??J_cg8`{$qGaz^AF_ z4DK$FN)F(teFz~@PLfKqXAs!+PKlmAodIC2?orxkz2|gjYAMAhau-r8kai|WB#sdH zk9kq(VdN0yb|ThX*Cm@B-cD_7fh23q{md?pkS`siwDOQjd5kx!lDks}0T#O|*sg^} zvC`O7+3YYQx!;kFq~+^&EI|jfH6ZpG zN!^>46jwHcbW0Cx@?Hc;0L;wg=-b^-q0+*&3Zl3jNc?H&WROlOTE3=KsfKFx4`~0F zI)D^F@4yhBryX)jbp)W}8`1f;rUL^>uE(1|)%61ZYlA&zyVt z+tiJJd5A632XpO5nJHcjZt<_ zWGsO=Kx}zyJEnmk=GsF4Hq)$ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_new-tab.png b/Media/Themes/Umami/Icon/actions/stock_new-tab.png new file mode 100644 index 0000000000000000000000000000000000000000..41f9ae0334dc2f8d25fe8397924ed57e897e04f4 GIT binary patch literal 791 zcmV+y1L*vTP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00J3FL_t(Y$L*6nXj4%X z$A9@MLCo`rw{SzE^wQ+Eaxk z;ePay3Jow>Gr_&w-KqBWjw5GIpWSdWV+_*0rX}?ZFVTz8Ckcbkp~qN(s4~v>@Qx5) z?79gY?&$e*7g$_eWPW~ry%o;YK-zYGWt(0`d95hbhjL{d1v@+`D=Es##aa{u0fj;V zt!qLxFl*bW2Aw35Q&`_1_(oug%c-P$FVeG7EIQfgBR_Jc!|DhmjvNq8R-vV~$X z^R{Pi{^hTnSis-9UoXv4`4oO|7%jh}ye=kRy?c7S|E6gYU{#Up!Pp)$yEFLyn-{Ek zJpTlUTR4ChQ`b~L+`rWTqxVN17-O=H0)$uv4CjViGMRihaP3A`0~lkn{Z|Kwq6p{Q z@5Fkl7dOUm>EdOSQjAZGT>-LsZLm}-(TIHCkGBY*TCF0btUG>v-)m#m+FHQG<3|m3 zU)RpK!`p+mx>i<}m6ECx=s?hJLn}iYY8N4tl3{40Vj`hKYi%J6!hAZNwj;UGx&PAT zmKxGe^&8ymBB`fE0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jY zFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vgD=;uR VFfiv?T%-U1002ovPDHLkV1mfLL>B-6 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_new-text.png b/Media/Themes/Umami/Icon/actions/stock_new-text.png new file mode 100644 index 0000000000000000000000000000000000000000..f01576b5f17ce433947043de8819af206f7e4c21 GIT binary patch literal 851 zcmV-Z1FZasP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0Q>;y08cd6u?+wK010qNS#tmY4c7nw4c7reD4Tcy00LD>L_t(Y$F-EbYZOrw z#eetCY$OSOkWDf!f|Z4cl^`a9|A0*jD?72Z7C}rSDt`PE7B*rb+lwLy>V`B1wKChp zC2=8$tn2PhGS_0>d~7mnva4PgX67>QoOjN7ci=yl{0oq+jRgTpKc@5W-JH?U#gB#+YF9+vh8W z_iAfn;i(@~yMEEadcBV4d6*d@f=Dbh{|rV+8cgU5ywW%}b_wyOI5hnli=JH;`@Rf( z)WTpKPy%LVV1}7tkzod~jYs z%K!n%XCR=0Fowx7LdtA&it))3wUr;7IobFC{2HiWX}Q)PYEDghk*M-8oy5H3$k=IG z?Gk=aZ71_kNO11#9FwKDis2LlL9nND`>j5LD2nbpcy#a9-vEAHYh3Mf z;7{(1$YCR{8?Al=KB$&GFYmhv0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>% zI65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(b dH##vfD=;uRFfi7!MmYcg002ovPDHLkV1iNAT0j5* literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_new-window.png b/Media/Themes/Umami/Icon/actions/stock_new-window.png new file mode 100644 index 0000000000000000000000000000000000000000..0890e6034ea0874de8327e4c916d08cc7ccd39dd GIT binary patch literal 875 zcmV-x1C;!UP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00KlwL_t(Y$K{jHOH^?b z#Xt9rqs};kWo{xo z2508o)8f7FywREP!O*I^`TltCd(OS*+zbCWtX}}_4qk%d6xax851MOMZ7hfH1O0)Q zX0hSRFgOFzDEJOEK54fwZq@pH7s0EAm-m6WJm8h`Dl6;DoEZH=sZ_*l3~3Ae$Je33 z4YV7tShZ0*T*2Jzjhp$>Z@qxlJ5U7*S0OqBUSq#LKMh+m&^Qm>1W0i9)}tfwZ(n)O zoog#xICqkAWrA|$9<-igHbx?sm3w<2XZ+OZBMAmZezC7jV4JvAqrUfy;o|p1CVU69 z-jl#3UO&EjB!Ov~NXIGY4Hhi|pCR6+UT-o~ZG430hn@r?>0SLD8bin~B5e^z5UB?G zjYSh|Zi{1;r8%oM_IeYz6U$Smj8%YHaw~)YaUJt1VpcKg6ru^%HhwZa^WLzg9J(DX#?vPRKWU%b3)GcgJljX`lu+$U< z3qTz{DeNpCfh1y&GcXK1+e#S$CKSjlCjp)qEetay85%gr-rA-MnPC#1t~n)gJWW$- zqlNMkMd&zptnxe+h_=ax=fMop4n7d!KPPtup86WvF}1=oO6x%PaLQ9bm;+DcNem2g zVcnG}d1IEdO(jt(l@68P37Otln3K@AhWF6#070wSdbsp<@hSbqgFfH?Z}A5yXF~0} zpB;4o0013yMObu0Z*6U5Zgc=ca%Ew3Wn>_CX>@2HRA^-&M@dak?_?!z0000bbVXQn zWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vgD=;uRFfiv?T%-U1002ovPDHLkV1g2F BTJ!(_ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_not-spam.png b/Media/Themes/Umami/Icon/actions/stock_not-spam.png new file mode 100644 index 0000000000000000000000000000000000000000..3f89d1960476094884730e0e793c4e3a3a43cbed GIT binary patch literal 1395 zcmah}X;6~~5M30iKyhpxE6S*;f>Jg13m`(YPEk%Riii$TOF;~B3}7{gARb>x5Jc=Gp*mWSRpku=D!P#|8cmEH?dvWuAYmT)ae^ z-+nB9%s+sI(ODRsjWO)(5H?270Z0MBuz0^8e18TAkD%fl6z9;{ILhTR**Kb#!oYBJ zzPMcG2NKUX&rG<0;V3g97Q<0Cc4dCPkE1O1<*SKt7>-^|j7vfpl~SWoPfbl>INI9U+|whS(CaZAO-;R;ew}^$&QH0& z+E)Kn)FHng+|T5n{uwKcc53q@O(i$pCVQL9+oDi*hOcD8kPc1R@c5=n=2hlNzy zA(e`{y2RbP*K~Jx_Vh@4d!>DST{2m>T;8it$dpQXf4{uHU!hVd)oP_$-9Ip(9vsvR z4Gjzr4~~osX|=pUi;;j>n%5IK^%7O`qpW;i>vSPeCOp-r)iyw4sOls0Qm^f zuSM`Mc33kUXDtBMz#`(Y2f>uUi?Yla(Ng4_wU&)4^T%W5)gS#mFW>U=Ia~L;WnbS}XkFPS z50!mk8}?#GvV>u{(S1?+8aL8EOEHdXf+wLJKDN z1XBra9-dw)B32{-Sj`004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Of~L_t(Y$F-GDNK{c2 z#(($CpR%-3#~C#d0zt`Lm& z$dUj<9?LC-d@3cC0KaRxwEEUA(cD~Ll-=@jnALAH>iGJMRDdQhmDKJ$d$;>K9b?U9 zzVT5$>A%q82=2J~Y_zj&?BaB~XlZH9@JWhDuCW-buC|Garn8pLQwul2Fv=Dp3cDJph@W zDS;USXRQ;f#lq<5D2YU(D1g;!MF@ezQB5>1iryG2WlRWx&1Nfku9V7oIUqo8oZs)? zr#STVc=sh(RzYuXZ&B9CM-A=QW78;^TS7rLOiD>lk2jC8l*zuQ_rjs@;aikjD7Qr`D?c&KEuzsV zN*!j%MFC_=fnsUt2Yr3lGjX~eIV`|#UhzzeQml17f?*h#=H|aB$PRPMGzo=5sMOsz z41-{Bm5G-VTxpmq50D=3NxLMx7N)VWDRb6Tkf`gCyy8y#*1;7t%P)*Q93~pqC}%gC zK5@blFW;h90b)a!SvR*6uJfmLI@O1~Kt&l75HS;g3KeW;?e-xd0mQ9ruV(EY;ao7yg!G&p#(0;mRRfjYnixKn-=5Ct}XH6RT90@hROrq4IE|HpBZ z_6HEIhl|*D1{(kX03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eN zIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF)=GJFgh?W T*0Dx800000NkvXXu0mjfXQYbf literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_print-preview.png b/Media/Themes/Umami/Icon/actions/stock_print-preview.png new file mode 100644 index 0000000000000000000000000000000000000000..760d4c62798d9e72616d6b121c0b7289b059d187 GIT binary patch literal 1291 zcmV+m1@!ufP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00a$5L_t(Y$JLcvY?MV5 z$A9y++jhI#i>4R4BwVCzjH0QOLIRe?@Sv?cXhGCc#h941S*b+~QI?jpm`F|Z!50h> zE7TC9(2LlLAvLy@fTmmoxfEzsXzh(lPv_>5kdTNF0-Mc7a&mHLHl~5JnZ$QgY|lD% zx}D-;JMCSgd~~#rv)w*QOG;_{_An8V5!h@tG)?y1a!&uK`+taf^&VaG&76Zy00ds zsI1(Pc}IYhl$5J|ef`W&il(c_&B=~_nt$o$(Rmi4@DddvaJz@dTDoj%4cg1f9feFe z`jd8)l{*ThRH>&NKZ=x zs3|LVYzWm23l#@yDt7FouKo}vAz7ThkmFq=oWIn`khHS)k#LT;9wAH%z{i8pk~Gt4Y^%3G&Eoe4`=IJ zZ*$?&uN*#DgRirh>OCKlm%E0>rf*o1@fZmSiLA?CPu2Uo7#QdW6#MsAGdS2ksiu=U z`FeK`Jv}{GtybFGPqN8g!p)mb+K#odp`eh5A6ksx@29b;k&Q3Eg2iG5AT}ljzdwM> z<-+gRX9ox&P)f10vXX%A$1o&PN({rG1oVKPQpFpMgvsv)G4J1=mBndJb1mMXkh{&1VW5wS<^HkBTUo!U=l#rbrcG# z)k0)cBx7S^pumuFJaWSzKK_1`QYdL4l^kCyg_IJd1U{cHw7xPF@Nx6^M~j|%GLIwQ zHE}QYz`|5IyRP8#`I@Ey{DHuy5hhd7@UWYlRZk#%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vfD=;uRFfi7!MmYcg002ovPDHLkV1i-P BOG5wv literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_print.png b/Media/Themes/Umami/Icon/actions/stock_print.png new file mode 100644 index 0000000000000000000000000000000000000000..0c35a115626b7974612b3ef4eb2d47b3d15c8f1b GIT binary patch literal 1030 zcmV+h1o``kP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Rg~L_t(Y$JLe3Yg|S)8F6E;^IAWxg6PSmc_+;IF5tXdRMGH5!|@(EkO|A`+nk6 zQRGMNkq4jfFt z(-Ni6>_|{57e4~-r!#OI2gh+ZapGl$hlkNxC!VtV@!9UW#T%|qeEJyxjYb0iYc0kY z1_z(#`0+t}-^cSjHa0erJtKtz!1YqOcrlHuwgX>Zt1&x2&)n=R)|wsf=$PYBC=__@ z^;0~2_!qanzXkB|rHRj{9+yTQndf=(q~^M=xUMV9%ga(K7n6SIsG_xcjrqGjbMoY? z96EHErKKfCM@JEnL~mnEl9gCN=}pNeC%<8A?3IqdI|%+pT$-35j9M6DH$ep?o&+K`Ms8=(lkLG6 zOD>lqigqBV*XxXo6gV=_PkpV91|+Tn5h091{%JI5Ha+w{g;ENo+R|yQ>FvpEDS{4y zD2fn-e11Q@*pO^uAl6`|ouP@5{uayQ8g3<>D1!{Hemm`hdg0QQ$e?+1PEo#{vz& z1^xi4Koc#&X1Fth*w001R)MObuX zVRU6WV{&C-bY%cCFfuSLFgYzSHdHYDHjtgI}7Kp+$fMIw<{ zEG{T0kVqslnM^L1E0s!>N~Nw))zs89Ha2QBnx>{EtyZhk>2z({);3*xd%Iq*@9600 z?Ck97>gw(7?eFiWC~9C}z+f;84h|ZP#-X7hlgTtZJUlWoGCDdsHa0dsK0YxqVK$p5 zCnu+-rlzN-XJ%$*XJ_Z;=H}<;Ef&kd!h+RmU0htW*=$QoOLn{6;czT3FR!eutgfzZ zZf+iURG$pMOPZE)5l*hOz21idgTo`(oHKFp*`=k*RQ30E08ZKG5@NZvqLN?$V@a`` zbnioMAGj&|`Cjgn?e|++sBo@Uob4yMyV2Ggl*w~`xhKr8s&=S&d#cC8Y4EOyW_^L~ z4ElO~{1N|33*~1$*~3@3mxuPf_SbN#GZk;y${#EFJ3!aI(aJ}P>^5h*Phnlan%-z} z^ZHtS?axM}!`D~f=6ST{`}{i5Od3u zE5e72&>nmD3$}~T%|!99K#CpG^SdOB?b49;_-x*8is#Pubp6r>g)BhR5`0m7_4Bx` zxY7wwY<~I}-BI~q-$ww)q;Z#|xqN9JE9ZJ1Uz!Id!6X<&6oUv)CnH&5r&wh834&k| xM4c^?{3p0pm@CS^1tvj~3?iICMx>KW7BhlHlH0+6K0az|QBqt=Y-q*$ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_redo.png b/Media/Themes/Umami/Icon/actions/stock_redo.png new file mode 100644 index 0000000000000000000000000000000000000000..ab73631109595c31b33c4d71ca04bb3fe46740e7 GIT binary patch literal 960 zcmV;x13&zUP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 Mp+Vs!SpZe=FLHDF)c^nh literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_save.png b/Media/Themes/Umami/Icon/actions/stock_save.png new file mode 100644 index 0000000000000000000000000000000000000000..747686b650a599fcb879f9faa0468de8574a7100 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_text-strikethrough.png b/Media/Themes/Umami/Icon/actions/stock_text-strikethrough.png new file mode 100644 index 0000000000000000000000000000000000000000..9b0c115e905091c0c354644c02469c261c1acf7e GIT binary patch literal 948 zcmV;l155mgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00NOoL_t(Y$L*6}NK;W9 z$G_*^v%B4G`QJ^4q1Gjc>0Zjrg2EQoOAkF5Q7H8gBYFrTDOV6g=tB|oWY9xV7G(uN z6cq}B^`NGs#q}XIEla!VS~)i7?!O*#)O0$@pdS3fkHh)<`F+pt9AHKNT4X7U%8%Y@ zP!+8{-e)P(*x7Wz`&R(2nwu7j#W3D@tkRrmHDYoqiZe~^kw{p!`ES*RQ|)I`Eim$= zc2kLifDl}L(x+=WflIIvgNdnGii4B@%Fmu!`l3k8O@Z?(4)@6rA<^z_Uw@$?KRzX|}bVs-5M=U_?W@#Iu^NJ z=48vLcA*66MN!kl=%5EcQouY1uKics85ye{xwmh&2Bu=LXP9811e7x1SOSi+;3xx* zGUyNk7#@#cY{(zdVl#XFcaOKFbU-NMT~l0WdH*Q@UvH11>3UL^vv;o83Y|19E8qHU z{DT+3JV1g2SM|kWP%2hz-NuWeI2{a3Hi#`JE-VCa?z?=}oSx;)&KLO4@4BansyO}k z>xLF`Kw@&#$><#4(cTIIB1`@4EdcBQSp58y@sFti(8k245A)Vn)rEr+4}e!=kx~F3 z0jP_%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q zEipG#F)}(bH##vgD=;uRFfiv?T%-U102y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(G WZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00S~fL_t(Y$L*9)Xk0}c z$GA{;!VXIOb-D?q0rJ>X>P3` z(yGvl^wwPb0}_grgl0|Hy4k#auibt7=k3ee-FY+Pq1%!_ZfL#ep*_?(I2BX@yG+F3i1O*liAVw9gNQ2QtL+E{OZFK(SpseYSi#oz9jqN-6Yx-K!3!Kb60<|wBEkN z{Ok#@{UY1Q*7dOi$w;6l+=-0oAgh<~+wU2YGs@7dI_$bcPc#&;b$x92Q)r`rLTBE2 z^-yrdsG-gTmNTXe;JZJwrp-Ko)x3kgL@cOMI`h9cz+Dbxhjff2X<|P z`O5y&j_tB?bpz#E6V#^xk{>*F=1S_R01743Z@>0}rkCodI389rd6#?a;xE@vxtz0$ zYdOESv1lX!$mu5rj7(gP3*ntSKAfa?G8H&=2IGE_Ky6+E0I1DN4~iCX8w|{o z=nh8}Da6T|R!rWv502byZ3q*<=rul#~Pe6d8 zGZd1dNvHk^u*IgMqqC|#&}V-1!5h(>#h_btEG{iOg+ej7ec+yO#OX~wALLxZbvKaD zrmb++NX*aAZn5l^hgR&5jqTgp?FUh@ZdDQVs~CUlm7wNR5b!C`{1lpBLC3=g{_yiU z{}3=i!OEw9(mWD;(p3A5Y)7ZRS3;9P4 zMG{@i$jr0_42_>pDU=L$hV~$P-$kWb#nPSIo{+NbcR~p9dIk=wel3WOP#BgJgG1xz zQ%m1|a(f3rrOI^jxuj;48fbU|xwWjzh5T~y+Q(nC1H5$V!r7{|{#m^LPzONJ73tL~ zmOc&Oy&VAIv6s{7yGlBJw{`o38R=Ws03-k?t$1V#34rGpfBe4p#Wz27N@ho^T_KbX zxBAH)F52d3^iZ0hzmTT8pQsAOHXWC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfd};CU literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_text_center.png b/Media/Themes/Umami/Icon/actions/stock_text_center.png new file mode 100644 index 0000000000000000000000000000000000000000..896dfa106be7b47a9277f420976407891a146bfc GIT binary patch literal 694 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s8-o-U3d9>=#%KJ9nNL8R@xx%5qAScq0VFK0X#brQy(dBoX~nOL{IeD4u!q`iBTU^Ci^}ARR8>s zG4qG>jd{zzUy!xxOO|sAxx8ij>(#T}GBy;Kd$)JJt8I^oRrQM36Q60Uu5{vgzf-c8 z+u9ihf(#5tp9wLZKd--1VcJ!uMIsGlzk`&r_ZRxTWZ2zmKF!5(CBr7OnagJKCPeBk zlywmhP*Hevlc8th0X3E*8Y~SQw*QwhzH?G)s{3EM+O{o`<(!h;`i>*bYj5Uo=y>K? zGtMdY571%jJQT!GemKFP#Uya`RfkuqyG1416jvx|R{z(IWlVl?chcEo8yJ+_j_iCR z*|J%>*DZSO$}oMW^-+tMPxAHBojmYy7_4tXcQps&8M^+Am*ciCe^rTP#mjic?_{6`bCY zVSFs-*bRTZ>6=AEXJ*c_(iZFbR$=pRmfwDHg2D?NY%?P UN}v7C2ILF|Pgg&ebxsLQ0H3x3jQ{`u literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_text_indent.png b/Media/Themes/Umami/Icon/actions/stock_text_indent.png new file mode 100644 index 0000000000000000000000000000000000000000..6da223e27f1c96b515f761fbc7cefb4ec9ee7011 GIT binary patch literal 854 zcmV-c1F8IpP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00LM^L_t(Y$F)?=ZW2)# zJ$Lw7AcU$kK}HsKTc5-y&{bn=vGFG{rcm|`IB8>}F>0+3pnDVF$AJLSg{h&GB7Dci z%)K*|5m?}zOmb)L%sJmV=erm9&qL|S(VN3B!{PA$GC-5bA((7!-}LeN_Q#9tu+dhfE?>l1iW1O}W27Nh_GJ{-S~wB*rd$)n`- z+7tmp;Lz22#bc-R1Y}S#AP}vj^SFzX9gBqX}spJ=z+ zXm`B3)9#?t>7aMsGvC!;)lschQM8Njdp5F7BNQ-XkM8&A;W!Q&jRr&rQX)6nbOz>F z5ChJCY$5<~4jeR_&Cq^XAY>pYoU004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00NasL_t(Y$L*BcOH@%5 z#=m_ob7mMbFR6oNn77C@)6pC&kP3qWL&}WE2MNs^J_r%?2LwI&WRRdG5;=*KLW-oA z4pEk+gkhMJ=B2X18fP4J=A5&44{D^zczdV^KUi|4h;K=31Z5od2%sjr=KRno%+-~ffYwsF_%_(mlV7yWxGQHfA zny@?66df#7HTAnkC#D7gz&5>U03#a7&m0#Kn2>%rOmn~OgEBB;{iZVR7y$fmxItBA z;_GuqI8>X+{kJ}Wg9DQ}`>?BncOgh|P^l!kLUXiJ&zBIsFo1dE= zt80BT4olCFlc_AVyW-*~05Gb+YG3&Vs{%~L53=-H;YNN&ysmq|ik9aCGqNn7?5WJ| zU-VE_#`LTLrpoIBBo)@Bih@vGkP)x@_|<`j?R~SVqGt9~W_K-_j;g>VQ_%IaD6zmC zBXD?}b2?Tx?Ph3w`N<`_-8wMZ9G_DK#wfrj7^5JB%(-NyWovCwQc-OP$5=!wP_-_x63JK(En!25KyWQMdotNur zR?+8f&hH7vJOjKO69?Rc8AryxpsZ#LL{%)k4cD#%2m-)k{@*PA9l8)I2&|vqbq0Wt z?*JqK3IG-Gysn9d006cCV9R1l|CxRPe}pBR%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK! zF)c7QEipG#F)}(bH##vgD=;uRFfiv?T%-U102y>eSaefwW^{L9a%BK_cXuvnZfkR6 aVQ^(GZ*pgw?mQX*0000| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s6sPZ!4!kKLiF_Uv2{WI0`V#dUp^-QVIFTqX&hVB9^arE{BB;Lez7$}7UOv!6zk zyjee4rg2#&V}iMWz+n}JGaT!xBkI^!35jyGeq}ksvo$I-Xwn18i7hTJCiR-TkZfhK&LeuZ8|^XA@-R-BG?$)}Ul}tZHY` zPUo4N99>5zFP&6T#A5ybG0W;$A-1*?pS_w&tk$aZ*PCx-X-nK=YqPEIU+X@F%K6XZ zCY_agG4t=4xc`gudEJ#?A1_?L+w$X@$=PeX zwHdRH8BF!Lxi{{8vB3{1=R5j&mp^ig?Yf=7@?Gd20|SFRdP{kVo554 zk%5tcu7Rblfmw*5g_W_Xm7$rofq|8Q!L-j7hA0|x^HVa@DjAFn3=MS+%ybRSLkx|q zjLfYJjUXDHN5(8+U|^60*$|wcR#Ki=l*&+EUaps!mtCBkSdglhUz9%kosAR&0|SGn LtDnm{r-UW|CRhRb literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_text_left.png b/Media/Themes/Umami/Icon/actions/stock_text_left.png new file mode 100644 index 0000000000000000000000000000000000000000..224d70023345320156b7f57b56361d1cd02092be GIT binary patch literal 684 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s7XPZ!4!kKxye%Yt8=p+xlL(soX1_(z6=1HgdUd=gd@`Y8CYD zjPBN?LuVKmWU9A|T*%t1!W6Q4D#Ieq6<61p#GcnvVoJNf{(|$oQY1?z^CXuiyPjUy zYqlV0XN+WfiTr(U1_K^v!Fi%gNg_-RB?mgZmIf*8s?d{WIJ2fF)7Vq%pcm`u|i_Tee<#d z7haZZc{Jm`qx=D%0G+rsoh?dD(Ip$|ydE`mGTh7&E7NodndjV)8Y%C8{_N+uSDsHe zU!W`g^Uutg)@SE_*;jQg!J_0}&Hb4^YUfJ4G8bje>gb=Td6-8`sZa2tuv3LoNBm&} zo6AP*+j4o0UiMsnTekoA-}!<%vv1d4w@(q{O@E}6@REUnLAAs+q9i4;B-JXpC^fMp zmBGls$UxV?QrEyN#L&XZ*wo6| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s8do-U3d9>=$ap7uNBAhNeSqkE;mVZP~~ z%(GrF2_?0j(3cRlKiJ}-BDJ5Pxly&VbN&%0F}C2!kR8`qHA|v2WY?HyTFp&6xAXaz zGa?V>bMJM0z42yg>~|&3fWrn==GV7xGr75A|KlaAeqKo659i)=KJD^e>-02_<@`%5 zbN$x{7#K4!wBK_T+Hzap%jN2;rBaT2>$Wb={?F>LWQBrMMW*KjkC_ffmw7E+*toX(+sd#21r{&OkMm7C zG;Z$LbNmRmaA~^EPU?K6r;CYTW8|ddp{YO zVz1wyH!nK9-YI_ap7qD>EeNo<`04o1#Sf2PknvZ0*8h6e=bv|CbTpi|oDle^Xq-CJ z(fx>|l9dyOJIBYlUQ1e*kj)2?C{7#J8- zOI#yLQW8s2t&)pU6H8JVj0}tnbPX(Z4a`CeEv$@9tqjey4GgRd45odyFhtRio1c=I zR>@#wU}&goV5Vzm9%5)@Wn^w;Xav#lJThhp0|SF3$cEtjw370~qEv?R@^Zb*yzJuS c#DY}4{G#;P?`)(P7#J8lUHx3vIVCg!0Oe{Bs{jB1 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_text_underlined.png b/Media/Themes/Umami/Icon/actions/stock_text_underlined.png new file mode 100644 index 0000000000000000000000000000000000000000..fafd4fbe482ebdcf9817d499857b7ec8e8f72acd GIT binary patch literal 1038 zcmV+p1o8WcP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 bF)}MKFgh?W=UH5&00000NkvXXu0mjfY-sKY literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/stock_up.png b/Media/Themes/Umami/Icon/actions/stock_up.png new file mode 100644 index 0000000000000000000000000000000000000000..dc7c2166d39657f634cd86a3a84adf5beae9b633 GIT binary patch literal 1063 zcmV+?1laqDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/system-lock-screen.png b/Media/Themes/Umami/Icon/actions/system-lock-screen.png new file mode 100644 index 0000000000000000000000000000000000000000..f9ff34715ad12a2f26f514b67a62c1a44af5846c GIT binary patch literal 1169 zcmV;C1aA9@P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 Hl+XkK3f^zp literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/tab-new.png b/Media/Themes/Umami/Icon/actions/tab-new.png new file mode 100644 index 0000000000000000000000000000000000000000..41f9ae0334dc2f8d25fe8397924ed57e897e04f4 GIT binary patch literal 791 zcmV+y1L*vTP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00J3FL_t(Y$L*6nXj4%X z$A9@MLCo`rw{SzE^wQ+Eaxk z;ePay3Jow>Gr_&w-KqBWjw5GIpWSdWV+_*0rX}?ZFVTz8Ckcbkp~qN(s4~v>@Qx5) z?79gY?&$e*7g$_eWPW~ry%o;YK-zYGWt(0`d95hbhjL{d1v@+`D=Es##aa{u0fj;V zt!qLxFl*bW2Aw35Q&`_1_(oug%c-P$FVeG7EIQfgBR_Jc!|DhmjvNq8R-vV~$X z^R{Pi{^hTnSis-9UoXv4`4oO|7%jh}ye=kRy?c7S|E6gYU{#Up!Pp)$yEFLyn-{Ek zJpTlUTR4ChQ`b~L+`rWTqxVN17-O=H0)$uv4CjViGMRihaP3A`0~lkn{Z|Kwq6p{Q z@5Fkl7dOUm>EdOSQjAZGT>-LsZLm}-(TIHCkGBY*TCF0btUG>v-)m#m+FHQG<3|m3 zU)RpK!`p+mx>i<}m6ECx=s?hJLn}iYY8N4tl3{40Vj`hKYi%J6!hAZNwj;UGx&PAT zmKxGe^&8ymBB`fE0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jY zFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vgD=;uR VFfiv?T%-U1002ovPDHLkV1mfLL>B-6 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/tab_new.png b/Media/Themes/Umami/Icon/actions/tab_new.png new file mode 100644 index 0000000000000000000000000000000000000000..41f9ae0334dc2f8d25fe8397924ed57e897e04f4 GIT binary patch literal 791 zcmV+y1L*vTP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00J3FL_t(Y$L*6nXj4%X z$A9@MLCo`rw{SzE^wQ+Eaxk z;ePay3Jow>Gr_&w-KqBWjw5GIpWSdWV+_*0rX}?ZFVTz8Ckcbkp~qN(s4~v>@Qx5) z?79gY?&$e*7g$_eWPW~ry%o;YK-zYGWt(0`d95hbhjL{d1v@+`D=Es##aa{u0fj;V zt!qLxFl*bW2Aw35Q&`_1_(oug%c-P$FVeG7EIQfgBR_Jc!|DhmjvNq8R-vV~$X z^R{Pi{^hTnSis-9UoXv4`4oO|7%jh}ye=kRy?c7S|E6gYU{#Up!Pp)$yEFLyn-{Ek zJpTlUTR4ChQ`b~L+`rWTqxVN17-O=H0)$uv4CjViGMRihaP3A`0~lkn{Z|Kwq6p{Q z@5Fkl7dOUm>EdOSQjAZGT>-LsZLm}-(TIHCkGBY*TCF0btUG>v-)m#m+FHQG<3|m3 zU)RpK!`p+mx>i<}m6ECx=s?hJLn}iYY8N4tl3{40Vj`hKYi%J6!hAZNwj;UGx&PAT zmKxGe^&8ymBB`fE0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jY zFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vgD=;uR VFfiv?T%-U1002ovPDHLkV1mfLL>B-6 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/text_bold.png b/Media/Themes/Umami/Icon/actions/text_bold.png new file mode 100644 index 0000000000000000000000000000000000000000..87bfdb2c7ca50dcd52f718f6bc29454679204007 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00S~fL_t(Y$L*9)Xk0}c z$GA{;!VXIOb-D?q0rJ>X>P3` z(yGvl^wwPb0}_grgl0|Hy4k#auibt7=k3ee-FY+Pq1%!_ZfL#ep*_?(I2BX@yG+F3i1O*liAVw9gNQ2QtL+E{OZFK(SpseYSi#oz9jqN-6Yx-K!3!Kb60<|wBEkN z{Ok#@{UY1Q*7dOi$w;6l+=-0oAgh<~+wU2YGs@7dI_$bcPc#&;b$x92Q)r`rLTBE2 z^-yrdsG-gTmNTXe;JZJwrp-Ko)x3kgL@cOMI`h9cz+Dbxhjff2X<|P z`O5y&j_tB?bpz#E6V#^xk{>*F=1S_R01743Z@>0}rkCodI389rd6#?a;xE@vxtz0$ zYdOESv1lX!$mu5rj7(gP3*ntSKAfa?G8H&=2IGE_Ky6+E0I1DN4~iCX8w|{o z=nh8}Da6T|R!rWv502byZ3q*<=rul#~Pe6d8 zGZd1dNvHk^u*IgMqqC|#&}V-1!5h(>#h_btEG{iOg+ej7ec+yO#OX~wALLxZbvKaD zrmb++NX*aAZn5l^hgR&5jqTgp?FUh@ZdDQVs~CUlm7wNR5b!C`{1lpBLC3=g{_yiU z{}3=i!OEw9(mWD;(p3A5Y)7ZRS3;9P4 zMG{@i$jr0_42_>pDU=L$hV~$P-$kWb#nPSIo{+NbcR~p9dIk=wel3WOP#BgJgG1xz zQ%m1|a(f3rrOI^jxuj;48fbU|xwWjzh5T~y+Q(nC1H5$V!r7{|{#m^LPzONJ73tL~ zmOc&Oy&VAIv6s{7yGlBJw{`o38R=Ws03-k?t$1V#34rGpfBe4p#Wz27N@ho^T_KbX zxBAH)F52d3^iZ0hzmTT8pQsAOHXWC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$5000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfd};CU literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/text_italic.png b/Media/Themes/Umami/Icon/actions/text_italic.png new file mode 100644 index 0000000000000000000000000000000000000000..cded0616b95a19998054647d36e725b66440a1cb GIT binary patch literal 952 zcmV;p14sOcP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 bF)}MKFgh?W=UH5&00000NkvXXu0mjfY-sKY literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/up.png b/Media/Themes/Umami/Icon/actions/up.png new file mode 100644 index 0000000000000000000000000000000000000000..dc7c2166d39657f634cd86a3a84adf5beae9b633 GIT binary patch literal 1063 zcmV+?1laqDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RU`L_t(Y$L*9$Xq;6P z$A9NOCYdI&C~a(|v07Ve35sZ}WJatasVHqNXrw+?TH2*FAw^d%bSFM=At(xVLfTM+MWW}j^)C^uvuy}Et% z>pQmXY1^=Fvp@aYFSlIVQ@?n&5RUz0K>mR5Y0I{M^jP0BZ70r+FnOs+@0JHMCr_W) z)_HgQ&3GY<{%yd$19TbRQt02l_o`C0NU{7Ke@vglI>+rB^O>K|{II(#XNSi|;_^}& zX6c;M5u81`>%ON~23upad>m&S)_6`%9itML=*#WwK%>uk4{*&D0s8x~iu+{y*2lX# zSFJHWO?`zmK5+*kd^`3jZS5;0e@9=Z@$Sf-`|wQzCT_HE=eqm0c5mwOKTdps6^nPC zmP`hpF{rI$yz=$fFzeTCaJO!}t2?{W9%`xvxj{F$<%XW;9?I{@e*f#|L{h^#!8j17 zRN^w&7;6PgVDj=Py_+9)=P#VUsr|MKVZ0E&OzX}2XuSh=$NDv!K74lP^I7LSNt{)F@3CnT{{b1}p5kc?b=P2B_9+3?KR6&r*5(iILahj~klvKAN#Icy{ts_QiMi z&R)+;^`(99wfc;sQlDhTR`A{+fGQX#y!rmi)u^tmiNp$=3;>G*MqY{TUyv>}*5Wgs z%e5(di@_K`1XaaaA*yR@;nir#ma}ZDFh;0EzoRz7Wd!2{5kZ3@cKKGeJU}efa1k+J ztl+%CI3cPBq={C!7(kNOHTS}*Q3V7vK!_Dp(0I1cbGe5&@n|W406tlB$sAR9pld%W z0b@Z#W+N3sm?Lx~!ZXR(n5$a=vng>0l!o4TYSl7XC^Zg~GHu)_okdw7?iWB!;>r05 ziASl_%mDSoLql{iOV`1s#2a-9DM*}x=)B7O8}=7g7!4tze#7Jd001R)MObuXVRU6W zV{&C-bY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Ny!L_t(Y$JLZgNK{c2 z$A9O{luTMW10mr@21*D*s~{}WjUau2MQx%MEex`-7D*5-S{UODEejF~S_Fk)i$2Yw zXj9Qvl0Xsdi!$>gC1)JxwRrcvc{9%B1h(kFedpeH@B9DHIro1a{O=~tBpm6y2DGja zZuhy4^T*TPgd?5SN?AD4X*sz5OzN$zx6;;6x|*}(EeHZwG<7^m+j`r5IBl)Da#R%; zT+BldL~Ljg5D}h6ccCie#K{G|Hn55_S6?HdAc~-%hCtLEN5d~zQ{cz%ll%xopfB9a zhERa&vSRMN8nvTUP;m(CjBY<4qOK}LX)r*4bPv8PnJgjbkL*T7xzfIaiuFa51_QkR zFk?rI*!34-=+g|O3fZgXyBOdTDku7nT^EW;PD~IV$7h+Ii_`mZI8`#qL#7RG(MbkIeo#{pbOD@b zD@%(R7@6SB;8*rvcnHAulRe85SQa$2_cHl2#>kf$P950dmd%l{(+fwMY8e=r;QRD1 z>f4rrXC(kgS`b@U5xAmY-bD^mwG(?E9?Zlm2 zpZg$-Of!GC%h?~!OBz8Uk+{(ny>#ra&~sQk7QdOs;z=n06dLb0CG>md&HzZ7u>d5D zj|20-91sJNnZoq~iU&|U-sd5(fJKi_8oyxX|K5HBKC0%a9W)b90000rbVXQnLvL+u zWo~o;C}SxgGB7YSAWC6tbz&e>bY*F7WpWA%Lvm$dbY)~9cWHEJAXI2&AV*0}P>|V+ zZvX%QC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK z03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$50000< KMNUMnLSTXk_mFk~ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/view-refresh.png b/Media/Themes/Umami/Icon/actions/view-refresh.png new file mode 100644 index 0000000000000000000000000000000000000000..1af6789278d35af0a65ff002585889d85ea410b1 GIT binary patch literal 1495 zcmV;|1t|K7P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 BTJ!(_ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/window_fullscreen.png b/Media/Themes/Umami/Icon/actions/window_fullscreen.png new file mode 100644 index 0000000000000000000000000000000000000000..db5035a681d82248ef14d1106372d4e1fddaf2d0 GIT binary patch literal 988 zcmV<210(#2P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Ny!L_t(Y$JLZgNK{c2 z$A9O{luTMW10mr@21*D*s~{}WjUau2MQx%MEex`-7D*5-S{UODEejF~S_Fk)i$2Yw zXj9Qvl0Xsdi!$>gC1)JxwRrcvc{9%B1h(kFedpeH@B9DHIro1a{O=~tBpm6y2DGja zZuhy4^T*TPgd?5SN?AD4X*sz5OzN$zx6;;6x|*}(EeHZwG<7^m+j`r5IBl)Da#R%; zT+BldL~Ljg5D}h6ccCie#K{G|Hn55_S6?HdAc~-%hCtLEN5d~zQ{cz%ll%xopfB9a zhERa&vSRMN8nvTUP;m(CjBY<4qOK}LX)r*4bPv8PnJgjbkL*T7xzfIaiuFa51_QkR zFk?rI*!34-=+g|O3fZgXyBOdTDku7nT^EW;PD~IV$7h+Ii_`mZI8`#qL#7RG(MbkIeo#{pbOD@b zD@%(R7@6SB;8*rvcnHAulRe85SQa$2_cHl2#>kf$P950dmd%l{(+fwMY8e=r;QRD1 z>f4rrXC(kgS`b@U5xAmY-bD^mwG(?E9?Zlm2 zpZg$-Of!GC%h?~!OBz8Uk+{(ny>#ra&~sQk7QdOs;z=n06dLb0CG>md&HzZ7u>d5D zj|20-91sJNnZoq~iU&|U-sd5(fJKi_8oyxX|K5HBKC0%a9W)b90000rbVXQnLvL+u zWo~o;C}SxgGB7YSAWC6tbz&e>bY*F7WpWA%Lvm$dbY)~9cWHEJAXI2&AV*0}P>|V+ zZvX%QC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK z03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGAl4JIxsNjSzM$50000< KMNUMnLSTXk_mFk~ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/window_new.png b/Media/Themes/Umami/Icon/actions/window_new.png new file mode 100644 index 0000000000000000000000000000000000000000..0890e6034ea0874de8327e4c916d08cc7ccd39dd GIT binary patch literal 875 zcmV-x1C;!UP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00KlwL_t(Y$K{jHOH^?b z#Xt9rqs};kWo{xo z2508o)8f7FywREP!O*I^`TltCd(OS*+zbCWtX}}_4qk%d6xax851MOMZ7hfH1O0)Q zX0hSRFgOFzDEJOEK54fwZq@pH7s0EAm-m6WJm8h`Dl6;DoEZH=sZ_*l3~3Ae$Je33 z4YV7tShZ0*T*2Jzjhp$>Z@qxlJ5U7*S0OqBUSq#LKMh+m&^Qm>1W0i9)}tfwZ(n)O zoog#xICqkAWrA|$9<-igHbx?sm3w<2XZ+OZBMAmZezC7jV4JvAqrUfy;o|p1CVU69 z-jl#3UO&EjB!Ov~NXIGY4Hhi|pCR6+UT-o~ZG430hn@r?>0SLD8bin~B5e^z5UB?G zjYSh|Zi{1;r8%oM_IeYz6U$Smj8%YHaw~)YaUJt1VpcKg6ru^%HhwZa^WLzg9J(DX#?vPRKWU%b3)GcgJljX`lu+$U< z3qTz{DeNpCfh1y&GcXK1+e#S$CKSjlCjp)qEetay85%gr-rA-MnPC#1t~n)gJWW$- zqlNMkMd&zptnxe+h_=ax=fMop4n7d!KPPtup86WvF}1=oO6x%PaLQ9bm;+DcNem2g zVcnG}d1IEdO(jt(l@68P37Otln3K@AhWF6#070wSdbsp<@hSbqgFfH?Z}A5yXF~0} zpB;4o0013yMObu0Z*6U5Zgc=ca%Ew3Wn>_CX>@2HRA^-&M@dak?_?!z0000bbVXQn zWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vgD=;uRFfiv?T%-U1002ovPDHLkV1g2F BTJ!(_ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/actions/xfce-system-lock.png b/Media/Themes/Umami/Icon/actions/xfce-system-lock.png new file mode 100644 index 0000000000000000000000000000000000000000..f9ff34715ad12a2f26f514b67a62c1a44af5846c GIT binary patch literal 1169 zcmV;C1aA9@P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv04M-y0O^(@b;$q#010qNS#tmY4c7nw4c7reD4Tcy00WarL_t(Y$JLcfXk1ko zfWJHU&LhvJ22%}~g0+fbVp347n6?oVL~t-To_}E{&CSiL^y>h*+$H+@y0KQURuHjRYnS)dT0~@p z&H^wxn&sX1dI1<8AHOvKpjZr+jfhBfU6H98mmMv2Y->v5c@B$qf`pG`0z!t*$Gmq z2=}(u(z$0B&m8FC&+G+Gjh^A^)hifdXlmNd-lv}A;43e4?AVVOt+xW$>f5snYS~fC zzNfpH`g@8KKaX&3>em%czE4C3pDN7Vtw6-iP_~dfe zsIT{EsBdIsPg3Oc&_Ug9&0;?LnY_6fk=Z|J5@zS8Z3PJ z_QxDv=Rtc%JK2fsDqY8mXF2d%4mV~%iij;>On@~$M*C>(qjeEgE~26W#o`P<4G;Ij z27vdE_9F69rN2;_-0V)cJqL8TUQ;q_Xbx)v8C4p7~ zKF?$_hc=4IY7hlP2pSO$*wwSz{j^yV{Xl48C)AU-$kx{-LFE zH~=Yuy$)sxtFB|p1yoohDhHIy0ZPY-*Z5Rh)W%r^#DV0@%(dSNg{6Z@3rGlN3eP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WgtL_t(Y$F-GPXk1kk zhQED1Gt=COHnv_$#G+P<6ui7>f{GN1ZG%{-1fLWycu6Z(h58bzV3mRoR$Ct|LL-FK zw>}6G4N~+$G{uq@Okzv5BynL_n6=H z`qBL*TwR*vop|-ry<0hZae>c1K1D7cQV3(LQ8j1K)p0xTzVtZHzSI9Ca9|lhHkaOT zdyjJ-=Nz$f3=EF&#KU)TVZMHIMFva5c`!z! zl`4`5_U?Xw_186UdUTTGU!4JbnILu!7drw|30IZC7>(AlOmK5sBOP0kx;NAX419YI zJh-J9#ICAr6$ApXMxnJT6XfzSVdOAc5=JqBHF#f=C;^Y8ThWtA$chyqFh(cCYOK{1 zV$W+I4U)@;^d9@3z#0OpNd*RL4c2O`(NzhmOQaRr=;Wys>A(;>kI{;QFLV+ZMLvuf z9-ZQgug?-oK^zyV;a8VPYlSI}=7uzHb>C0#=RZ)0Vm|2qiCiHn%VVu3iUeU8(^zN! zn;?_2EXK)cjnTZg^Bx9R)+mhC z1S!q#&Rh9$G|QF6h}MQQo7-DaQX;O>MdeI{Ah84?kU}s#mc?4l%iHh5<4IY?cO!rD z>!mAbCHds!2!%qJSjqA}(m+%KPF3qBk+5-ke42B=&tr{32#F00Qpn;#iXS6Z-h&jd zAS(z2WUMTk8Qzy~tp}tML{XIZk5Bx_dytj#q?Sq3Y%Wh+oSbWW?A~^U$Fc}PG3^*4 zM;I4~784i6dz=UFJRXm7o+ygg_Rww2&d#tdm>OTnR=2Zj`>w4$$N%>ar)yuI@9cj2 zAW&D5x|B&%$5Z>?FpX_b&qYnwvx>OEPM)8iogEnJJMbPb2SiJi(?!-4kXdm@^uJ(6 zE9QR#i@-u@uO@%0k>F};*Dn46oWt2%*+|cc0000bbVXQnWMOn=I%9HWVRU5xGB7eQ zEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q lEipG#F)}(bH##vhD=;uRFfc(bR}cUI002ovPDHLkV1n(6{k8xA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/accessibility-directory.png b/Media/Themes/Umami/Icon/apps/accessibility-directory.png new file mode 100644 index 0000000000000000000000000000000000000000..29cd13746bbf9c46313920008e7cc6c72a060ca4 GIT binary patch literal 1171 zcmV;E1Z?|>P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WgtL_t(Y$F-GPXk1kk zhQED1Gt=COHnv_$#G+P<6ui7>f{GN1ZG%{-1fLWycu6Z(h58bzV3mRoR$Ct|LL-FK zw>}6G4N~+$G{uq@Okzv5BynL_n6=H z`qBL*TwR*vop|-ry<0hZae>c1K1D7cQV3(LQ8j1K)p0xTzVtZHzSI9Ca9|lhHkaOT zdyjJ-=Nz$f3=EF&#KU)TVZMHIMFva5c`!z! zl`4`5_U?Xw_186UdUTTGU!4JbnILu!7drw|30IZC7>(AlOmK5sBOP0kx;NAX419YI zJh-J9#ICAr6$ApXMxnJT6XfzSVdOAc5=JqBHF#f=C;^Y8ThWtA$chyqFh(cCYOK{1 zV$W+I4U)@;^d9@3z#0OpNd*RL4c2O`(NzhmOQaRr=;Wys>A(;>kI{;QFLV+ZMLvuf z9-ZQgug?-oK^zyV;a8VPYlSI}=7uzHb>C0#=RZ)0Vm|2qiCiHn%VVu3iUeU8(^zN! zn;?_2EXK)cjnTZg^Bx9R)+mhC z1S!q#&Rh9$G|QF6h}MQQo7-DaQX;O>MdeI{Ah84?kU}s#mc?4l%iHh5<4IY?cO!rD z>!mAbCHds!2!%qJSjqA}(m+%KPF3qBk+5-ke42B=&tr{32#F00Qpn;#iXS6Z-h&jd zAS(z2WUMTk8Qzy~tp}tML{XIZk5Bx_dytj#q?Sq3Y%Wh+oSbWW?A~^U$Fc}PG3^*4 zM;I4~784i6dz=UFJRXm7o+ygg_Rww2&d#tdm>OTnR=2Zj`>w4$$N%>ar)yuI@9cj2 zAW&D5x|B&%$5Z>?FpX_b&qYnwvx>OEPM)8iogEnJJMbPb2SiJi(?!-4kXdm@^uJ(6 zE9QR#i@-u@uO@%0k>F};*Dn46oWt2%*+|cc0000bbVXQnWMOn=I%9HWVRU5xGB7eQ zEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q lEipG#F)}(bH##vhD=;uRFfc(bR}cUI002ovPDHLkV1n(6{k8xA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/accessories-calculator.png b/Media/Themes/Umami/Icon/apps/accessories-calculator.png new file mode 100644 index 0000000000000000000000000000000000000000..92afcdbb4bc70475d13c9092981c6a8bfc307157 GIT binary patch literal 977 zcmaLWTTD|29LMp00Vnr)VUyrS)QPZ|;%Sl6mO0bXO3Q?uo=eZA7kY<6&;r$!LQ5?z zy_i~12eozJE}2!>gpw>~X32sxyJRk1c5%rT!?Mg7_FyuyJ(y)ICqC_EpXB@ad;G=F zuJ(%3nojB_9su41dYn+g|H4TPMxgB~B8V!bNa00_H z41a>a2@)qsfRX;pYKec-|rtE9}jTO zaKgc0FgU>pO+-$HLZNUt9F7f!W8p|75{*VNw^lYcH|6tdasa4wV_i7#o+{m0*VH1G^qugI()1%40PrlYv?~Jk zkLG#-c!$&q8PR}SO?wJmzrR`77!4T>h9ko4!Z#D@Z@>R7D6?sg%TM_DIfb~onvRu4 z-s&I%GAH9?w&X&+HRqohu{66%UatB$IaR!UM^)8Yq+t6%NU?a-pjGrlE3vJbv~$;mz1w~}_-%x=5XI%GGy?I1*i2p^H~5fLE}OPiXd0?|PPks`>004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00QetL_t(Y$JLY1Z`4*0 z$3HXAYdf1{vk3%P0Z}W_Gol31oZ)YXiiFxjk5uZRlvFJs1XPW9~A#cva+(AOCT4Z zudm2z^A$!7h#(>vr&5IUZmlJ@5&eBbgq`rs!RGZEiH2J&_T{$I8U)yq#3Tg6=0Kssv!&m`ileX+jAiI z(9j_4A5Hf54)3GcS|bbtO#1I+dd6JAPjgiH@C4| ze|hKi8LU6JfB$9_<5yq2m=vLJHvn-Q2laaWm%I07XMlD1?>p4gGcQfe)VI|;^X#PA z&eY#u{0m9~_&+ex{m=ja03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`If zHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*7SL XFgh?WK`vJi00000NkvXXu0mjfRA{wM literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/accessories-text-editor.png b/Media/Themes/Umami/Icon/apps/accessories-text-editor.png new file mode 100644 index 0000000000000000000000000000000000000000..7ed1d9916ba77df7b761d6aed37575395a693140 GIT binary patch literal 1007 zcmb7@dobGv9KgRpEf`DBx;kTHT`eBTexZ#t=jPEwrS09^W+Q;f9k z&dARIaEDloP?UJK=*yXN!y$xl&ePM<^v8o7DKY%uG0D=oShOM|tXwR5^{HKO{#V8* zGA$=C>u-{hs7$S_+8K#p%{LUWz9n*+f_sM(3l61J8Xhw3J#2~sj8S8qcet&M^j3_QRNnHznIT_-(t1>8zOr7tx`kT%Vk6xY-d|fj7 zUhPJ3jZTcG%=OJ#_qoPkzghC$zA6l?xkx6lH%6AZMZPs5wRL5uUu<7J@}^JBF+odO zp4xPy_q?i|z`D1jC8xh;T6Z%w|HKd9`E^cjXLx^m{^6F=wItIysM2SGD5};l&Mdyv ze&jjv$4#WYnkYHnEYWy?O`#(2^`2AadqS?uYc`olr@$TK^4tK-=Z zu3_G<1QMnnH8I&X0|1U<%w#Mh4dc=|*STpJ7yJ=_goKeZOLPwCA xO&aAt0Z))|DYF3l5ekKbP)U?Pl;ThK52RB9{tZIONre5+Sj=d~cVVfe{{Tvgs0{!B literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/background.png b/Media/Themes/Umami/Icon/apps/background.png new file mode 100644 index 0000000000000000000000000000000000000000..fe80dcee3060418d8fc581705cefc75733b16e85 GIT binary patch literal 1170 zcmV;D1a13?P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WdsL_t(Y$JLZgXk1kk z$A5PwnR&@`CQW9{Xw8^3LSsyAZ8a_gwSpyxU_eSiK_Q@47p^KH;!05vckYY}H`ax= z(hioo>Bf&VZ8bKLG&a+;O)`~srkzaZ^L@YTVqQXIrkPOOc;N8fJNKURyZ>`|7yj=i zR$o5-?31^E@&Bmq!LiW~z65Am7aSiR9!61G4bwE8+8l|LfRsoi(glg@#Xw4l?buRL z0%{_W7zb(q037-DXfTt>0B|N*BCiQ<<_)Bjb&ncrYhvH7F!6YtXlFZ-NaSDZHGoVe z!^p@8uIo}ca)v;&pOe2PDQiYuK-Prz2lsF-b)KOYo(EuNX6DWS0A1ISQnED`;qmQJ z=2AKm!KGBLE@WaZ$3O_Crlx8X`06E?rip2qs6ihs0f@E*80_yv2tbG$znCf@CG6h4 zhn}9Ejavj@Sr$@Cgpf$r;r`}*zH^9Mxgv{K7ILSsNWo%UuRP*s&ss4BEK zw!;3g11K$9Sa%5k;JPkCNG2yvQEmhX27~x~)uKX(wPy+F+wl}lEj=|k($xW&oH(_y zg08MErl+Ut%I}VVS(w8qFVq64|EJdgUVLe&dM(Iml9Y1oJAM5s-yi#KZSUYf5X18E zXHr{>ZR;CkcA-=k@I@k#c$-G%?*~A2{-T_}PDumkpzOtG7fRT+{nhGrafdQ3KzMZQ zy~$^uepbD@csXhqhR-z3YC9x^nP>130X~J_W;Nb@|8O^u_k80a1I<7%oz6@Yi-lMA z?tRX;va&*1dCLWOCjF0004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00eVML_t(Y$K94|Oj~6b z$7j^UOH7Q3lBfxOm@%2g#F&_UF`9W<7B2yu1EzRO+!Eb%7c2vrXrtg55;wN8ZjAeF z1zTF6&y zSJ46XIhp+aZ=;S$N-MMEmaAcQOh<$*%m_lE5F}DLR_=?LSwW+*<(Ss!2+>T3%%V>qHf5HE;vTVVK6i385l!eRcFLxTk#}L zDgR8F8qUcX^bR->4u{cWvcVM$!#T4GW@;H4%Q)J5%;?aWFu$;f`T2S1h9;18S+RAx z*n+V+Wm6IgtwqFJ+d5!%2jQ9r+{B3QCQK98;hFYgd3iaqW@TjsQ&S#DdK^euugT_x z|L|B5Kfk&YbBpU6g)J(K&jAy`e{lFi(DpMBG%8WA=tI3!4WEAoUauGZ!%lFowy_4> zO$|OeRpt?QSn-$F4`Gu6rpcQaBLkN4C^qjZ4AIn0(>k@zf=0CwKA#T*LkzA+jI6=t zwMfaV40R8!$2voyZT@7t8n4O(PufYgTb|Gro7$l#n zSlGNpzbp0CDhH7(WMCh4t#1y@p`^ux@{UOqt6V6QjpA%YJGjM4boUKnad8oDw;TQa zCL|o^fhRJsZ=l=MAmPX*QBJ7}yk=%Yglx+Sq+R$;Y(q{xgCDBRNWE$RuT}w%#{;X? zio6TOIGQiTX};q2T@-hy;AOUtr8+0%8hvCm+%lN|I@nY z8=MBbexeS0Q)_Yfj0EMfL2xMY_@fuSF|oUnX#WeugIgL9FA?wV`z+g@l37hdOo>8K zAF7lzBGemK{u?wmb|-w($h zxwxF1R<=eh2%5NSabFi)e=XtUAVqza)_p;|6AkR4J3V(EC3X?767R9cL{#W7aVQ#b zBC(Hng?OB}=Wb5U`-q2$$A~A1SmLRu;3LHS|M@?MzX7|+vv@@3Z=?VK03~!qSaf7z zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6W zZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*7SLFgh?WK`vJi00000NkvXXu0mjfQqg1P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/calc.png b/Media/Themes/Umami/Icon/apps/calc.png new file mode 100644 index 0000000000000000000000000000000000000000..92afcdbb4bc70475d13c9092981c6a8bfc307157 GIT binary patch literal 977 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 F69ALPcg+9* literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/email.png b/Media/Themes/Umami/Icon/apps/email.png new file mode 100644 index 0000000000000000000000000000000000000000..1187e8dca1ada3fa9b77cd88bed9cb18eb5dc192 GIT binary patch literal 930 zcmaizZBW_;9LIlbwA)Sz8SAFmMeffFyxo-7dIRO9?3rV>Y0N5Mf*3AH*?|U)<0N9VSOR@n3o!g_d7XSc9Cs9fOE*%7DxeO2n zfEG7EBLXmA2|)Z2U{C!G$;rb2u&tg{@fkp(K0ZuN%Fh=8Fa_jdY99Pwu3l^Xch7%% zuD+p`dAa%Owbst>zV5wyr;mNNkHcbfSZpq`6$zU`ZjV7bXWHOu0W{cTku~@8T ztJP|?S#36(-TvHew@*(`I~)#&(=jtM<8(S*E|+W8H9I>yH#g^YyWJkQ$K#owpZ9vb z3kwTApU?02FD@=FEiElCFR!eu1OkCzFc=DjR##Wo*4Eb7*Tdm(Boc{6qp?_QV`HOQ z-5D3Qi$N(ZgH6~iTX(*+FXfH2;u7*n%J)5sHEWKzAIzDeg52{;43Y(a$tcXtrlve? z)p*L7$DxfTCAa&1Lv>tbkGV%H626JUVX2$iIkAmn;^PCYgiu5jI7(Ph?!YcTFSmF4 z+>g6dAGHNBpL9!l^Y6UU_(7NK-M_>^*z0UFd!r|4m}k3xD3VPQZpoRF{o7m=w0Y{b z@|%~Nx3^@`KCCFux>z$+`i|;KtQdVPl0?GlmBV#e9nf}p)5u48E%=I1kc4k1>LFn$KvGg@fzl|K&(ZTMiL`vv%Wn=Oc z{IRqm%^tG*NHBEd^r@tZ>y%wgz_>OiRyJzU=%IEdnRE*GOCk0G+`fTyB#9a@%*>=5 z#;dBYaKc}|w3qF^Nx~O5eeq&fv1-JHEtH)R@K#kn?cWI39>wH~E2ipcye0s6jgeQ* zI7?$x5$O$8G)5KR5j>KLyq$`qQ_)N!;VmMX{yKsX5#)Lx6a5cRTX*(c_2+;`P&5@u gPen7RD4vMVAfkkSfb6vQcE+z!m{*$1%c*4k4L_aFmH+?% literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/file-manager.png b/Media/Themes/Umami/Icon/apps/file-manager.png new file mode 100644 index 0000000000000000000000000000000000000000..3597ca1e1642608a57b12124375aa8640f659374 GIT binary patch literal 559 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;JpsQj7}P}D;O9U@&kNATp5S}=}BtA0V+ZM${EROIqBNPX-ajO3I}TiFKlnRw5#>% zo{sDLdv0wly}hmM?$K%Yk4?LOeA=V^O^**WJw846@tJu~KxqEUi_8B1|L?o{wIc%q z1Aj@7UoZm`3oARHgru~Rs(S2#m5c9gsbXMYU`+CMcTw5G_H!l!1H%GO7sn8d;JJNo z`I-!PT(_U=^x4w!Iw0i3-ZkyKFN^;2cqTj6AG@!-_z~lz zTbt0^cV6|*0hMzuF4NX*ou+d!Qz+>D206}|e}8y2=y@e?Gd?z@wJmcR%hD@ZZknt? zH4k|!8zg_59#x#jz`&qd;u=wsl30>zm0XmXSdz+MWME{VYhbBsU>0I%VP$M;WoV{t zU|?lnFzvI2A&Q3F{FKbJN(LhXLqlByGhIXT5JMv?BXcW5V~7Svy>J0gJbAkMxvXn84}{Bih)h`DTsI$?TUrW@*9t|OHV zMbFC@J8sA(u3dUNDkJrdXxf#hCt2rcaiFK60dSWBFjN4q)btpNjsrks4g1VRfGA5im~6885&)*7@RT42{#S~2xuRX(-f^w7 zOIcWyU&Jrq6`WGE%Ns69FV@un00x7>VzD?J4v)u2M@PrS#l^?RCnO{!CMJ@}O--#Wt%`QJN~Kb()jd5u8ja?bR@2wlH!v`u)9D5W2fx#QtJmv?hK7cRhet+6 z3c5ZHN{^7%gg@wh%#l@wt z_hnzQFD=>aPhnrSFYhBP$MUknv5)X|IGs-CKAf(wT`rf~<#xH;ZuiQ{%8JM1*@xHb zU0q#WTU%RSU-$WZe!o8u2yAR@Y;JCDZEXdE!R_tsP$;ysv$MOqySKO3UwiTk05~M& z=f4Noi06ohV&fB%($X2I@bsD6hOyl&zdHQ$@;nwwXN@Xm05I=!P^KXEFz%;muAnRy z{&rmXJ!9flOp@Z))|kPzej#%=qFEvQ>tyhcl<%vpbfpdZ5~aHt&(*$sl|;W8x_9#+ zKvwBo0cPw;C{q65Q*$J!yjPA3i*Fox)!r|b=+54CxOMM#yM7_(op9?#Z{EJ&b%Azg zj34xRM!dSInwo(h2$5D@LABEE^ZRF8ECCE*!*22P)xLH|bN0xSy24(QRTG*KRK^^; zi+h3V3z;?x)@W%9k886(@AxXl=T9E${JB%zn{|CspYtB~d)*1=%-Bvlcfa#$rgi2RSbi ziM`AN0J6Qq`pT(#*0Jz(-jL=LODoExV#c`(;xefiXb25SLEcG0=mKgcBkgSlmHrlj zFc9Q=Ad~u^!pEOfR7o0vhES;~2t9?GA)wM2v3ip literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/gnome-calculator.png b/Media/Themes/Umami/Icon/apps/gnome-calculator.png new file mode 100644 index 0000000000000000000000000000000000000000..92afcdbb4bc70475d13c9092981c6a8bfc307157 GIT binary patch literal 977 zcmaLWTTD|29LMp00Vnr)VUyrS)QPZ|;%Sl6mO0bXO3Q?uo=eZA7kY<6&;r$!LQ5?z zy_i~12eozJE}2!>gpw>~X32sxyJRk1c5%rT!?Mg7_FyuyJ(y)ICqC_EpXB@ad;G=F zuJ(%3nojB_9su41dYn+g|H4TPMxgB~B8V!bNa00_H z41a>a2@)qsfRX;pYKec-|rtE9}jTO zaKgc0FgU>pO+-$HLZNUt9F7f!W8p|75{*VNw^lYcH|6tdasa4wV_i7#o+{m0*VH1G^qugI()1%40PrlYv?~Jk zkLG#-c!$&q8PR}SO?wJmzrR`77!4T>h9ko4!Z#D@Z@>R7D6?sg%TM_DIfb~onvRu4 z-s&I%GAH9?w&X&+HRqohu{66%UatB$IaR!UM^)8Yq+t6%NU?a-pjGrlE3vJbv~$;mz1w~}_-%x=5XI%GGy?I1*i2p^H~5fLE}OPiXd0?|PPks`>004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00QetL_t(Y$JLY1Z`4*0 z$3HXAYdf1{vk3%P0Z}W_Gol31oZ)YXiiFxjk5uZRlvFJs1XPW9~A#cva+(AOCT4Z zudm2z^A$!7h#(>vr&5IUZmlJ@5&eBbgq`rs!RGZEiH2J&_T{$I8U)yq#3Tg6=0Kssv!&m`ileX+jAiI z(9j_4A5Hf54)3GcS|bbtO#1I+dd6JAPjgiH@C4| ze|hKi8LU6JfB$9_<5yq2m=vLJHvn-Q2laaWm%I07XMlD1?>p4gGcQfe)VI|;^X#PA z&eY#u{0m9~_&+ex{m=ja03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`If zHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*7SL XFgh?WK`vJi00000NkvXXu0mjfRA{wM literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/gnome-help.png b/Media/Themes/Umami/Icon/apps/gnome-help.png new file mode 100644 index 0000000000000000000000000000000000000000..45c3c9318833af07f9fc5e8c651365ccbe6c849f GIT binary patch literal 1484 zcmV;-1vC1IP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm0000;0000;07l7cJ^%m!24YJ`L;(K){{a7>y{D4^00gN?L_t(Y$IX^`P}F4@ z$D!$t#%ao_Y^J#WVWgVO6swt{(UQvn3?V{FaJ*nBzyNhogn?Z~4v|aFmGJrKV~6&3tBO_ucpVeV_01{C>L{8vmC|+m!z_ zv5C+pH2*Eptwbu(OZ-ku6SKq|(M*I98~*8OLz$Sln-Kh@N=W3LxHdS>QB>0fesU$u z-IAH<6k-2�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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/gnome-session.png b/Media/Themes/Umami/Icon/apps/gnome-session.png new file mode 100644 index 0000000000000000000000000000000000000000..32b3e87d7e9047ab5cd14d3a4a0d011252eda6e3 GIT binary patch literal 1139 zcmV-(1dRKMP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMza_)tt#MF0Q*000000000000000h>C@0X=NWth95|T zA4r2xQchA-P+?BaU|e2WU}9aQq@$jmot~bZpP!zs zt*jtQhcsoRHD#kVXQeo1qBv=$I%}vqY^XeFp+0Y`KyIi&aH~Ueu0(8|M|ZJPfwxsx zQ&?J7Sb(%yTUT0KSX+d)TwYpTc9&g?z+PZnUx&G2V_#usXJL=SVv4+DdXi;lV`YAv zWs16Glg4I_zG;rUX_v`rb9iczz-y4cY>~Kal*4XsY;JFDZg6aFkF;)^%x{yxZp2~T8cX^Jfe0_O*qRoAMdw!+Rfq{R4f`5Xj z(S(G9hpg0ziiwPxrjU}3lC;>Am6Vv6n4qAbprN0kp`fLvq^72(r>CZ5uCK4J zva_(YwY0Uiv$nRjx3{;rxVX8wxw^W#yu7@j&CAWr&Ck!z(9qD) z(b3e@)YR3~*V*0J*x1=_>=;`a}>FMg~ z>g((4?d|RD?(gpI?(gsK@9^>P@bK~R@$&NW^Yiob^Yrxe^!4@i_V)Jo_xJhv`TP6( z{QUg={r>;||F3er!~g&Q8+1}mQvd-10|Y5if{>J-vCq-e+1ugn@bUBX`~3X;&Y*6Z z00001VoOIvr@*%>0{{R4Pf0{UR47w*(dRpYQ5eVZCwuSRF|s*GBqMw8z4tCFk0`PV zk(II&Ndy0;>zvbbiRYMS!c7*bRzlNER8=rzD zCJ+@&$WH?pCy9eqGoc6!kdmB=)ia?K*fEfvfweQC3YamFMRYTv4m@L-z0e`p6Ng5i zN7Q8-uNIr_@$N!76zPRR7dX{uAcTAN&rie2^6W(Oeop`jec1p2_fj52~#yh z2ZXS*KaZA3BqYqVIpb~7G)=++F^6v}9#P0-G90`eyYcV!I-QPtpi-%h*5U8MEZ)3Y zv9rCxT|2ux77Cn#TOdDdDJD8DHY!vQ%I65;mIx#jYFfckWFqZQ;F#rGnC3Hnt zbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(bR}cUI002ovPDHLk FV1nvdCe#1` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/gnome-settings-accessibility-technologies.png b/Media/Themes/Umami/Icon/apps/gnome-settings-accessibility-technologies.png new file mode 100644 index 0000000000000000000000000000000000000000..29cd13746bbf9c46313920008e7cc6c72a060ca4 GIT binary patch literal 1171 zcmV;E1Z?|>P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WgtL_t(Y$F-GPXk1kk zhQED1Gt=COHnv_$#G+P<6ui7>f{GN1ZG%{-1fLWycu6Z(h58bzV3mRoR$Ct|LL-FK zw>}6G4N~+$G{uq@Okzv5BynL_n6=H z`qBL*TwR*vop|-ry<0hZae>c1K1D7cQV3(LQ8j1K)p0xTzVtZHzSI9Ca9|lhHkaOT zdyjJ-=Nz$f3=EF&#KU)TVZMHIMFva5c`!z! zl`4`5_U?Xw_186UdUTTGU!4JbnILu!7drw|30IZC7>(AlOmK5sBOP0kx;NAX419YI zJh-J9#ICAr6$ApXMxnJT6XfzSVdOAc5=JqBHF#f=C;^Y8ThWtA$chyqFh(cCYOK{1 zV$W+I4U)@;^d9@3z#0OpNd*RL4c2O`(NzhmOQaRr=;Wys>A(;>kI{;QFLV+ZMLvuf z9-ZQgug?-oK^zyV;a8VPYlSI}=7uzHb>C0#=RZ)0Vm|2qiCiHn%VVu3iUeU8(^zN! zn;?_2EXK)cjnTZg^Bx9R)+mhC z1S!q#&Rh9$G|QF6h}MQQo7-DaQX;O>MdeI{Ah84?kU}s#mc?4l%iHh5<4IY?cO!rD z>!mAbCHds!2!%qJSjqA}(m+%KPF3qBk+5-ke42B=&tr{32#F00Qpn;#iXS6Z-h&jd zAS(z2WUMTk8Qzy~tp}tML{XIZk5Bx_dytj#q?Sq3Y%Wh+oSbWW?A~^U$Fc}PG3^*4 zM;I4~784i6dz=UFJRXm7o+ygg_Rww2&d#tdm>OTnR=2Zj`>w4$$N%>ar)yuI@9cj2 zAW&D5x|B&%$5Z>?FpX_b&qYnwvx>OEPM)8iogEnJJMbPb2SiJi(?!-4kXdm@^uJ(6 zE9QR#i@-u@uO@%0k>F};*Dn46oWt2%*+|cc0000bbVXQnWMOn=I%9HWVRU5xGB7eQ zEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q lEipG#F)}(bH##vhD=;uRFfc(bR}cUI002ovPDHLkV1n(6{k8xA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/gnome-settings-background.png b/Media/Themes/Umami/Icon/apps/gnome-settings-background.png new file mode 100644 index 0000000000000000000000000000000000000000..fe80dcee3060418d8fc581705cefc75733b16e85 GIT binary patch literal 1170 zcmV;D1a13?P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WdsL_t(Y$JLZgXk1kk z$A5PwnR&@`CQW9{Xw8^3LSsyAZ8a_gwSpyxU_eSiK_Q@47p^KH;!05vckYY}H`ax= z(hioo>Bf&VZ8bKLG&a+;O)`~srkzaZ^L@YTVqQXIrkPOOc;N8fJNKURyZ>`|7yj=i zR$o5-?31^E@&Bmq!LiW~z65Am7aSiR9!61G4bwE8+8l|LfRsoi(glg@#Xw4l?buRL z0%{_W7zb(q037-DXfTt>0B|N*BCiQ<<_)Bjb&ncrYhvH7F!6YtXlFZ-NaSDZHGoVe z!^p@8uIo}ca)v;&pOe2PDQiYuK-Prz2lsF-b)KOYo(EuNX6DWS0A1ISQnED`;qmQJ z=2AKm!KGBLE@WaZ$3O_Crlx8X`06E?rip2qs6ihs0f@E*80_yv2tbG$znCf@CG6h4 zhn}9Ejavj@Sr$@Cgpf$r;r`}*zH^9Mxgv{K7ILSsNWo%UuRP*s&ss4BEK zw!;3g11K$9Sa%5k;JPkCNG2yvQEmhX27~x~)uKX(wPy+F+wl}lEj=|k($xW&oH(_y zg08MErl+Ut%I}VVS(w8qFVq64|EJdgUVLe&dM(Iml9Y1oJAM5s-yi#KZSUYf5X18E zXHr{>ZR;CkcA-=k@I@k#c$-G%?*~A2{-T_}PDumkpzOtG7fRT+{nhGrafdQ3KzMZQ zy~$^uepbD@csXhqhR-z3YC9x^nP>130X~J_W;Nb@|8O^u_k80a1I<7%oz6@Yi-lMA z?tRX;va&*1dCLWOCjF0n84}{Bih)h`DTsI$?TUrW@*9t|OHV zMbFC@J8sA(u3dUNDkJrdXxf#hCt2rcaiFK60dSWBFjN4q)btpNjsrks4g1VRfGA5im~6885&)*7@RT42{#S~2xuRX(-f^w7 zOIcWyU&Jrq6`WGE%Ns69FV@un00x7>VzD?J4v)u2M@PrS#l^?RCnO{!CMJ@}O--#Wt%`QJN~Kb()jd5u8ja?bR@2wlH!v`u)9D5W2fx#QtJmv?hK7cRhet+6 z3c5ZHN{^7%gg@wh%#l@wt z_hnzQFD=>aPhnrSFYhBP$MUknv5)X|IGs-CKAf(wT`rf~<#xH;ZuiQ{%8JM1*@xHb zU0q#WTU%RSU-$WZe!o8u2yAR@Y;JCDZEXdE!R_tsP$;ysv$MOqySKO3UwiTk05~M& z=f4Noi06ohV&fB%($X2I@bsD6hOyl&zdHQ$@;nwwXN@Xm05I=!P^KXEFz%;muAnRy z{&rmXJ!9flOp@Z))|kPzej#%=qFEvQ>tyhcl<%vpbfpdZ5~aHt&(*$sl|;W8x_9#+ zKvwBo0cPw;C{q65Q*$J!yjPA3i*Fox)!r|b=+54CxOMM#yM7_(op9?#Z{EJ&b%Azg zj34xRM!dSInwo(h2$5D@LABEE^ZRF8ECCE*!*22P)xLH|bN0xSy24(QRTG*KRK^^; zi+h3V3z;?x)@W%9k886(@AxXl=T9E${JB%zn{|CspYtB~d)*1=%-Bvlcfa#$rgi2RSbi ziM`AN0J6Qq`pT(#*0Jz(-jL=LODoExV#c`(;xefiXb25SLEcG0=mKgcBkgSlmHrlj zFc9Q=Ad~u^!pEOfR7o0vhES;~2t9?GA)wM2v3ip literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/gnome-settings-keybindings.png b/Media/Themes/Umami/Icon/apps/gnome-settings-keybindings.png new file mode 100644 index 0000000000000000000000000000000000000000..017ef9bbb7ad331c276d8804e6ee5a488b5edcd5 GIT binary patch literal 1163 zcmV;61a$j}P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00U@AL_t(Y$F-EpiyT!H zhre_0t*)w>Oi0M2eIQOIiory16A@P~ML`4p0}_{t8yDhivI}8EL;`UmxDeck3vn@o zC@zGAfv6x7d?p!Wz#-17yQ*LHxW`3xPr9djf}?m=eXD=x_uX@<&Vm0qWbosicQ2d; z-uK=g+)HeXNw57bpZ>gNUdnQRXw>0u1lF_=&i(TFTNfDuG;ZC#saxBd+H5wo)k?L~ zX=}IJ(RQb;X`1Nj>L2>V%ZvWIAAi&JjSb!0-qJ?1sY#k@+G=T%rn=o|=&!d|^|3RH zEdVzFc<&F+%*^oDoi!9ic3sU_%j|5OwCiQ=!9#3q-odLM*BMLwXq_;yG6LvB5CI!F zjIoGFzXwX*%PEQifU+!66kBuHUAnB(+tG~yiXx*HMyM#(T0{&Yf~sOv$n&hP%L!`{ zY7}EE#ti8qL!M`->Np_Fa^g59FEgyIZifh}9>mejdVO6+9L40m(haIr^@s`R_PU_` zmJI_+UlK<###pSeSmy|wBd`Hs7?Jk@MOhHVQKc~!XDgkvjyRqo%X%miDo|BK46}#p zW1n8yN=DA=;m8DCr`sMmyWN82>sKcaj0ndbp5HlN|9*L|abbS$_&7jBk~B$X=8rwH zXJV2h#5K{i^+fKa+iG^maz6%m^U9Z3-c!|?y%I$veY7rLyn6nt7xQ9oowJfzBNt)9 zpdWkTLde7gaPL@ITIlxkF9Xj3&#x>kTy{go&qg)#Q7`v%`*l|^C(bNxW`)0`$}|1P zexyZHp}n;5%IV{$AHNSH`$MQIKdo#YdHaKF=XzD|KLfrV0knJNsi#ifj}^mR|6!00 z84jC!KshmYm|EbpmuH^OojAPT9)Awd)pL;9 zRk*tR4c6LyeMMzia&)$i4sI|BFcsP#mT&y|004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00VAGL_t(Y$IX;oXj^3% z$A9NN=j5YFX3ME7rCC#|WizcxEt{*#&>N9WF9zO>b_ibhiCzv@@v?SK^v1j47}YTl zKd-7&9HlrUda)Xu);ToF>eAYxNt2x9oV@4tqG_6>iQ0kS1LxwL=Xw9X=lwqq2ma&G zwtJxO;MCi|c#HJe)XedtwI7TdXlnfa(cM4Y67B4hWf!OHQE()ee_fM5et&fwr~~}# z8}uEVItz>eJbCm%-gxbGT&XG&io-|7`E>Rh0NVWe#CIPan*tEq=zQ$a!w(P%`dL`a zvTy%BeBqn%MRs!Q?R!~~VPgF;9)Ii*g_3-^r9~%BPF6du509+hkI)4xnIf^lQC>Rq z5V{a7{!wH$nZ`7AO0KFq)occ1cMJ`?{r&xh5CWwXfV%i)LXe&z67EDpBbCmO`XWI` zInU~@-T3_hG+j_E6q^HHd39=1DQ^NG9*>)eM50c*(vi3_J zeRUDt&LjK*zRJV?yN9rCyADvDfl}VYGfzE7Z?q2~1U)@HNGWj~hpw(JmY0`_Mq@nt z^z%f6*D=$J2myxAz`v4a_|5_Bg1rS$XOUqTSeC`o(h{*)jFFKMq?Dx7X|7!Pf!xDcbAS_;p$X00svKNhXuz^Lggx=1@w}8;g<6W*8p63tiXQPSmZb zloHeQkrb>43$ngZ4bZqjL4pg^kTzN?Li zI+FHq007evtpRljAqcibP|BlVmz&l|&qLER1!%JulQkJ{p2|phpwxp zHL-yxfcEWtiqy>UM}XgeYrvoXcmp8>hGC$25?#nG`Jb;ZJOP{n$Q9F*z}1Ry4k!a^ z3&1c8y1TnEjICVDvhewQ)iwS6(ns%diT5|gQ4w_j7pU%5RY@r|clyk!1Dh}t%j(2+ z9oBM7oSA-e(_zp11X!yGYa89x3RkUec-Mg)z|IpVCl?Z5&)cC;$Q+6f7%v=qaWC*Y zunH8Ki8Y;NGy4G(@B_Y@&dNZk(SjQRH0Ek8Q2gKGFB0BSTjtApe*gdgC3HntbYx+4 zWjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hi yZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGb=DKIxsLnE>{o$0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Xs2L_t(Y$F-GNXkAql zhQD+7zEf^Zz=%boAc~c$IMEjoTZ%~4BBBo#Y8AwiJ~)7a;$uXLhy{`8lk`m~_z(y5 z#X8{B5I;k%T=A9R;o;B%^z~PV1_o|= zamzzntCebXbsD;McB+DEt5sB$rteq3{qE~sW0PYI;B^ZSNAaGAAKX$sF@B8k@rjl& zJTDXJftSr)s(<^2+t~iZlasfyJr)UxOD&)z$FO)#u!wDx85DhB`dmO0WB*s?@`c|^^f*`jxmNX2yo)a z2@;nQSZk@*7dZ6Gu{LncK>-9IGrnydoQs*8oyS^B5Cn+m3V~}87(=aA;nq6_a!_f@ zP*qSguNvQ9qT$ce^c&z}Oi{4b;0JzQ2&sZ3m1C@B{^BA>4o~Enrye_0K*@lE2Fv~u z;-pmr05q2(jIsEBGvA+#AkHi@l&d9%-rmQ--$wX*=2QwQsU?l(A`6W=rJ|#JO;6iN zN;{>pO%*iPQlN^lmZgPd&YhoO*E6s3?}f9xzwaHU&z&I%n`t6BU>82nT#m4|rJ#)< zj$@n?tT7ZzhL88ZhqXlppMQf-KRLj*9rseJ6oF!gCG7(bD3%Py8WQJndou+Iagwqx zE;QKr^dOIH-pSqBx2%)?#ddfg5h&*(-N=Kr`FKgR6)=Ty9nZ#D2lMw5CsX(?tH4#*_1^TBv@;4F6p$$ zNkSNfSYs)e0>$!**Js8q&>b7@zp+bE`YIR?&+~9jx-Ciw!w|3Ffx`Kzdj3Jrx-yp= zUcpNbp>v%CB0>_!%*>o+u_`sseU z$Pa?U$4CF%+g@Y+z_mbcM(VM;I4`*c*^>Z^KpmKCfs}w^8#q@<&}tJP2BIs(xbpZ9 z1fW<*2_qG{0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckW zFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(b RR}cUI002ovPDHLkV1miz7Bv6> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/gnome-window-manager.png b/Media/Themes/Umami/Icon/apps/gnome-window-manager.png new file mode 100644 index 0000000000000000000000000000000000000000..2a02a2e16c9052ee769755512ecc4154a6191bd2 GIT binary patch literal 714 zcmY+5do0ue0LQ=La9*8kxq03^PR-7KcR2i#9O~|tJC9CW3CCR?8;aARX-7EzqmIn0Y3i@`+2_B{rzTd+x3zMz0$?i?M8yN3?`UZb zU|Mx>4S-oDPs9VB)mv}LC;%84f_M>7$^_Jvzy<*IEKo!Mb7=tC063m$?2ZcsV4i-I zABF85nqRbN08q~eq=kUB!;U7n*+>UW?g*oJ62aht=7rcfj&@{Zo~#lbt^S~o3ulyP zjaI2Hd0g2SLwg@7GRarVy*yjKQ;ip^n)!F@(t~t11B<6pJ5&o4U)R*{okdHHH$qa1 zhbjccbem0gwI=iH)WnxhA-P2b#QbKVKYg_#$vMD%O-JUb@hplD-;CkY%Kas$~O7IA2VW-_W1FV`HFa4mY<+R~ivKJN8qOZK;-GY0JV8QEpr&gxD&vHPTae1)4y;sXT z8)K*qzQvF(3VYW^2Y-7`KTl4ylZSKZtNi`zVWST`;~&52*>E(d&DFcM)n}ziZ%4X} zwdF3=&iM>_?B@ z;CDXSaBQQ-p$Gi8RyagxGE1PDI@c9a}{|?L>4GyN`;)E%2sAE zBv}duUnq)Ql6&U?@ZJgKGNppd_*;QPa0nI=!Xmg5b};E5NV2)U1VIwS_2pppe?xA* oJo|JBI0T!`BDgGekc7=4IYA`b|KGs#OQZpS@US@QNkm%3ANV2_W&i*H literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/gucharmap.png b/Media/Themes/Umami/Icon/apps/gucharmap.png new file mode 100644 index 0000000000000000000000000000000000000000..f99d513846ff8b4a2329cf58e937c523ce1a2832 GIT binary patch literal 1001 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00QetL_t(Y$JLY1Z`4*0 z$3HXAYdf1{vk3%P0Z}W_Gol31oZ)YXiiFxjk5uZRlvFJs1XPW9~A#cva+(AOCT4Z zudm2z^A$!7h#(>vr&5IUZmlJ@5&eBbgq`rs!RGZEiH2J&_T{$I8U)yq#3Tg6=0Kssv!&m`ileX+jAiI z(9j_4A5Hf54)3GcS|bbtO#1I+dd6JAPjgiH@C4| ze|hKi8LU6JfB$9_<5yq2m=vLJHvn-Q2laaWm%I07XMlD1?>p4gGcQfe)VI|;^X#PA z&eY#u{0m9~_&+ex{m=ja03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`If zHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*7SL XFgh?WK`vJi00000NkvXXu0mjfRA{wM literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/help-browser.png b/Media/Themes/Umami/Icon/apps/help-browser.png new file mode 100644 index 0000000000000000000000000000000000000000..45c3c9318833af07f9fc5e8c651365ccbe6c849f GIT binary patch literal 1484 zcmV;-1vC1IP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm0000;0000;07l7cJ^%m!24YJ`L;(K){{a7>y{D4^00gN?L_t(Y$IX^`P}F4@ z$D!$t#%ao_Y^J#WVWgVO6swt{(UQvn3?V{FaJ*nBzyNhogn?Z~4v|aFmGJrKV~6&3tBO_ucpVeV_01{C>L{8vmC|+m!z_ zv5C+pH2*Eptwbu(OZ-ku6SKq|(M*I98~*8OLz$Sln-Kh@N=W3LxHdS>QB>0fesU$u z-IAH<6k-2�`k623BWgQwp0fcxMVCX$yQJOR&p56_y_9@QslnyZl#-jLjfb(hLKu zkQv%jL_GgE@n?vBQ`h9>{#)b7E^EUF4pDg5HUjT|9fd89F);e}JhttN!-r0B_}oPZ z>7~nDRTge33$!=kaR%WA)*&jk!BtRzxA&Zd)9E}C>BhkDBo-EzFr!*P&y7)>%dW+C zGB7@z4FAL`GMQkNu>Kf7s-dd%qGpK8l$`ixhiEQ}hh5nf9WZmxz|QZ|QC`=J%u*$E z%=jauqqUkrx}|#xZrz^3R>x>;-W!dNoZ}%TgVNewoQS=M0|E&$%RAHtqmwFpc{~?Z z-r2C@{m8wy_Do|{*sf;K7m_9iYb?u~W&GJ;A&52ytpDKV$XsOy@a%LWm z0fqE=KDS={OE1o5DVAB(E5)BBmda~;;1y9!jf};o2a~wcnYb-44}1L!v0qS(mdnFx zgPOJh96VD7C%%+hkI8Gq#l}A9nhTm&%YS-kbecvn4Il51#}3z2SRBpbPR5A`m!Pq0 zNG*PScnaQeKjZtzD!7EnxOMNCN*e2F)`z*31{>K;HB57Dc~r!m-h@tX?UMt0zXCY( zi&3cPRf}I6o<@ME5j}I49|XAV-OlsrBF`DKI`g z51SuED$?4yn!yHQCco-8C{pssTWg*DyF6i?$r zALq2-_|Bei6`Ll%glet_-%t+@g_L1r{DHxFX(vUcB=J`Gr8ht%>!S5p814>bH<`{O zpGP!eGCQBlMK%pmnjEJ<37l#ESxlZWavV!kz&E81rzn;HQ3L8}bZ=92HeMMj9W%j^ zwLGr1=!+>-$!rllDcQ+NYOf&fLMu)tT|z)wEz--oAaA?LWiD0pEF0Q{&8|uOFPGM9 z+YZ_4m%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(bR}cUI02y>eSaefwW^{L9 ma%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000li~28?%a4Oj92WSUgEVxVC!;A{mp05}(bOC;bm0@6Mp#X2-q zS_mNCsw%$-2t1oa6^RBN0ESAA8Wi~7a&vQmKTi+@0I^v7S27xn)iox2ovn>&b+xrz zZgkW)*t@!~Go3DHOVjZ1Fw^O}>T+@%$8@^hyq#TKT=aUqKA+F;_Xh$2mSs7PTUuHQ z27}AX%b`$cWo2b`b#;AxJraq;%#6(e$g=6QdPK#<^0vgq^S23viQAKtQ&Q8?GqUC7 zRc4FT(ZpQq9S{3Va~=RuRT`xn3@<|k;IUdRGl=w0Z%hi^WPYl62U7YzJW4z%{GPen;a>9}G5(mSpFKS_f45O- zy?0i!Yva@Odv=;*Pq_m;_VZfs$t`DM?Z^Jf(f*o4g<3`FNCg`%-FZ{530R*=UIW-; zQyjCID{U63>4K%wW&usoBtae^NQr?ekmlq`Dak&Pl#*mmsDS!MICIW?qPiY5Nl^qT fA*g%?DV6hEknJx^*sQ2lXyp%!j4i(aK!fRv literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/internet-mail.png b/Media/Themes/Umami/Icon/apps/internet-mail.png new file mode 100644 index 0000000000000000000000000000000000000000..1187e8dca1ada3fa9b77cd88bed9cb18eb5dc192 GIT binary patch literal 930 zcmaizZBW_;9LIlbwA)Sz8SAFmMeffFyxo-7dIRO9?3rV>Y0N5Mf*3AH*?|U)<0N9VSOR@n3o!g_d7XSc9Cs9fOE*%7DxeO2n zfEG7EBLXmA2|)Z2U{C!G$;rb2u&tg{@fkp(K0ZuN%Fh=8Fa_jdY99Pwu3l^Xch7%% zuD+p`dAa%Owbst>zV5wyr;mNNkHcbfSZpq`6$zU`ZjV7bXWHOu0W{cTku~@8T ztJP|?S#36(-TvHew@*(`I~)#&(=jtM<8(S*E|+W8H9I>yH#g^YyWJkQ$K#owpZ9vb z3kwTApU?02FD@=FEiElCFR!eu1OkCzFc=DjR##Wo*4Eb7*Tdm(Boc{6qp?_QV`HOQ z-5D3Qi$N(ZgH6~iTX(*+FXfH2;u7*n%J)5sHEWKzAIzDeg52{;43Y(a$tcXtrlve? z)p*L7$DxfTCAa&1Lv>tbkGV%H626JUVX2$iIkAmn;^PCYgiu5jI7(Ph?!YcTFSmF4 z+>g6dAGHNBpL9!l^Y6UU_(7NK-M_>^*z0UFd!r|4m}k3xD3VPQZpoRF{o7m=w0Y{b z@|%~Nx3^@`KCCFux>z$+`i|;KtQdVPl0?GlmBV#e9nf}p)5u48E%=I1kc4k1>LFn$KvGg@fzl|K&(ZTMiL`vv%Wn=Oc z{IRqm%^tG*NHBEd^r@tZ>y%wgz_>OiRyJzU=%IEdnRE*GOCk0G+`fTyB#9a@%*>=5 z#;dBYaKc}|w3qF^Nx~O5eeq&fv1-JHEtH)R@K#kn?cWI39>wH~E2ipcye0s6jgeQ* zI7?$x5$O$8G)5KR5j>KLyq$`qQ_)N!;VmMX{yKsX5#)Lx6a5cRTX*(c_2+;`P&5@u gPen7RD4vMVAfkkSfb6vQcE+z!m{*$1%c*4k4L_aFmH+?% literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/internet-news-reader.png b/Media/Themes/Umami/Icon/apps/internet-news-reader.png new file mode 100644 index 0000000000000000000000000000000000000000..1b11937663f13eb04560798642b9565175b8f0eb GIT binary patch literal 706 zcma))T}V>_9L4`)rcR8|n37f(p`6<6-c2{R@?$!4vu5U2>f7da(-rfpE7Lt}b4)Q` zMx(`ul_NG~L51KnM2s;aX-4SQ2j-%dQbxmRjM7~qRuJ^of#2i#Jsi%p#)i6d%0UVM zqzeW7CIHBeladGkuU*fd0l=Pio`?r9X-XqyNdSPz1x+FV%`pJI5nvYp^q&FxP=GH| z0In4vtJ^kv{uBVDb_-gs0;FZcNmveQNdRy*o|V+X|3$4XfPOzM&ron zkjZ2+o6Q!B#cH+MY__qnF}vOFa5$V!=fuQ>%jI&r-P6<4Gcz-Db8{Y#$Lsa_e7?oS zMZZ7b4+Q?Qy1E(+z6*sy8yg$pa5xf)Y;A2tqtTt6omebJk|a6VoD-jhXlf8cVsa`a zGoN;x!7LH~=-HkGfP}U)b$l2}IwSx9w~)`1&|chLeoL*c${v>8I3RK<&*kCwTqWk% zsKFnp2ONG>)eZJDELh2DKy+0JDwS6r>=2uQG%nr1E7gY7z zZhgGY;fU*t*Z8ReDG{%x;#Gns{ZmN*6c=)`UlOAA_M)PoW|VwdxxIJkFgvSUze@#x z3Zk}!sOeIv@J<41dT|+nD-gAk=s+YLI8rARHBu-vc>%&eQ6^F^yVBQ&X+Y%a$9Q{WXh?~h+cSlhsVP%Sn50)Bi7ZU6uP literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/internet-web-browser.png b/Media/Themes/Umami/Icon/apps/internet-web-browser.png new file mode 100644 index 0000000000000000000000000000000000000000..05c92d7b5216c3a2ab8a9fc0ecc0f31cc9efda88 GIT binary patch literal 1393 zcmV-%1&;cOP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00eVML_t(Y$K94|Oj~6b z$7j^UOH7Q3lBfxOm@%2g#F&_UF`9W<7B2yu1EzRO+!Eb%7c2vrXrtg55;wN8ZjAeF z1zTF6&y zSJ46XIhp+aZ=;S$N-MMEmaAcQOh<$*%m_lE5F}DLR_=?LSwW+*<(Ss!2+>T3%%V>qHf5HE;vTVVK6i385l!eRcFLxTk#}L zDgR8F8qUcX^bR->4u{cWvcVM$!#T4GW@;H4%Q)J5%;?aWFu$;f`T2S1h9;18S+RAx z*n+V+Wm6IgtwqFJ+d5!%2jQ9r+{B3QCQK98;hFYgd3iaqW@TjsQ&S#DdK^euugT_x z|L|B5Kfk&YbBpU6g)J(K&jAy`e{lFi(DpMBG%8WA=tI3!4WEAoUauGZ!%lFowy_4> zO$|OeRpt?QSn-$F4`Gu6rpcQaBLkN4C^qjZ4AIn0(>k@zf=0CwKA#T*LkzA+jI6=t zwMfaV40R8!$2voyZT@7t8n4O(PufYgTb|Gro7$l#n zSlGNpzbp0CDhH7(WMCh4t#1y@p`^ux@{UOqt6V6QjpA%YJGjM4boUKnad8oDw;TQa zCL|o^fhRJsZ=l=MAmPX*QBJ7}yk=%Yglx+Sq+R$;Y(q{xgCDBRNWE$RuT}w%#{;X? zio6TOIGQiTX};q2T@-hy;AOUtr8+0%8hvCm+%lN|I@nY z8=MBbexeS0Q)_Yfj0EMfL2xMY_@fuSF|oUnX#WeugIgL9FA?wV`z+g@l37hdOo>8K zAF7lzBGemK{u?wmb|-w($h zxwxF1R<=eh2%5NSabFi)e=XtUAVqza)_p;|6AkR4J3V(EC3X?767R9cL{#W7aVQ#b zBC(Hng?OB}=Wb5U`-q2$$A~A1SmLRu;3LHS|M@?MzX7|+vv@@3Z=?VK03~!qSaf7z zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6W zZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*7SLFgh?WK`vJi00000NkvXXu0mjfQqg1P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/kcalc.png b/Media/Themes/Umami/Icon/apps/kcalc.png new file mode 100644 index 0000000000000000000000000000000000000000..92afcdbb4bc70475d13c9092981c6a8bfc307157 GIT binary patch literal 977 zcmaLWTTD|29LMp00Vnr)VUyrS)QPZ|;%Sl6mO0bXO3Q?uo=eZA7kY<6&;r$!LQ5?z zy_i~12eozJE}2!>gpw>~X32sxyJRk1c5%rT!?Mg7_FyuyJ(y)ICqC_EpXB@ad;G=F zuJ(%3nojB_9su41dYn+g|H4TPMxgB~B8V!bNa00_H z41a>a2@)qsfRX;pYKec-|rtE9}jTO zaKgc0FgU>pO+-$HLZNUt9F7f!W8p|75{*VNw^lYcH|6tdasa4wV_i7#o+{m0*VH1G^qugI()1%40PrlYv?~Jk zkLG#-c!$&q8PR}SO?wJmzrR`77!4T>h9ko4!Z#D@Z@>R7D6?sg%TM_DIfb~onvRu4 z-s&I%GAH9?w&X&+HRqohu{66%UatB$IaR!UM^)8Yq+t6%NU?a-pjGrlE3vJbv~$;mz1w~}_-%x=5XI%GGy?I1*i2p^H~5fLE}OPiXd0?|PPks`>004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00QetL_t(Y$JLY1Z`4*0 z$3HXAYdf1{vk3%P0Z}W_Gol31oZ)YXiiFxjk5uZRlvFJs1XPW9~A#cva+(AOCT4Z zudm2z^A$!7h#(>vr&5IUZmlJ@5&eBbgq`rs!RGZEiH2J&_T{$I8U)yq#3Tg6=0Kssv!&m`ileX+jAiI z(9j_4A5Hf54)3GcS|bbtO#1I+dd6JAPjgiH@C4| ze|hKi8LU6JfB$9_<5yq2m=vLJHvn-Q2laaWm%I07XMlD1?>p4gGcQfe)VI|;^X#PA z&eY#u{0m9~_&+ex{m=ja03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`If zHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*7SL XFgh?WK`vJi00000NkvXXu0mjfRA{wM literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/kcmkwm.png b/Media/Themes/Umami/Icon/apps/kcmkwm.png new file mode 100644 index 0000000000000000000000000000000000000000..2a02a2e16c9052ee769755512ecc4154a6191bd2 GIT binary patch literal 714 zcmY+5do0ue0LQ=La9*8kxq03^PR-7KcR2i#9O~|tJC9CW3CCR?8;aARX-7EzqmIn0Y3i@`+2_B{rzTd+x3zMz0$?i?M8yN3?`UZb zU|Mx>4S-oDPs9VB)mv}LC;%84f_M>7$^_Jvzy<*IEKo!Mb7=tC063m$?2ZcsV4i-I zABF85nqRbN08q~eq=kUB!;U7n*+>UW?g*oJ62aht=7rcfj&@{Zo~#lbt^S~o3ulyP zjaI2Hd0g2SLwg@7GRarVy*yjKQ;ip^n)!F@(t~t11B<6pJ5&o4U)R*{okdHHH$qa1 zhbjccbem0gwI=iH)WnxhA-P2b#QbKVKYg_#$vMD%O-JUb@hplD-;CkY%Kas$~O7IA2VW-_W1FV`HFa4mY<+R~ivKJN8qOZK;-GY0JV8QEpr&gxD&vHPTae1)4y;sXT z8)K*qzQvF(3VYW^2Y-7`KTl4ylZSKZtNi`zVWST`;~&52*>E(d&DFcM)n}ziZ%4X} zwdF3=&iM>_?B@ z;CDXSaBQQ-p$Gi8RyagxGE1PDI@c9a}{|?L>4GyN`;)E%2sAE zBv}duUnq)Ql6&U?@ZJgKGNppd_*;QPa0nI=!Xmg5b};E5NV2)U1VIwS_2pppe?xA* oJo|JBI0T!`BDgGekc7=4IYA`b|KGs#OQZpS@US@QNkm%3ANV2_W&i*H literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/kedit.png b/Media/Themes/Umami/Icon/apps/kedit.png new file mode 100644 index 0000000000000000000000000000000000000000..7ed1d9916ba77df7b761d6aed37575395a693140 GIT binary patch literal 1007 zcmb7@dobGv9KgRpEf`DBx;kTHT`eBTexZ#t=jPEwrS09^W+Q;f9k z&dARIaEDloP?UJK=*yXN!y$xl&ePM<^v8o7DKY%uG0D=oShOM|tXwR5^{HKO{#V8* zGA$=C>u-{hs7$S_+8K#p%{LUWz9n*+f_sM(3l61J8Xhw3J#2~sj8S8qcet&M^j3_QRNnHznIT_-(t1>8zOr7tx`kT%Vk6xY-d|fj7 zUhPJ3jZTcG%=OJ#_qoPkzghC$zA6l?xkx6lH%6AZMZPs5wRL5uUu<7J@}^JBF+odO zp4xPy_q?i|z`D1jC8xh;T6Z%w|HKd9`E^cjXLx^m{^6F=wItIysM2SGD5};l&Mdyv ze&jjv$4#WYnkYHnEYWy?O`#(2^`2AadqS?uYc`olr@$TK^4tK-=Z zu3_G<1QMnnH8I&X0|1U<%w#Mh4dc=|*STpJ7yJ=_goKeZOLPwCA xO&aAt0Z))|DYF3l5ekKbP)U?Pl;ThK52RB9{tZIONre5+Sj=d~cVVfe{{Tvgs0{!B literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/key_bindings.png b/Media/Themes/Umami/Icon/apps/key_bindings.png new file mode 100644 index 0000000000000000000000000000000000000000..017ef9bbb7ad331c276d8804e6ee5a488b5edcd5 GIT binary patch literal 1163 zcmV;61a$j}P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00U@AL_t(Y$F-EpiyT!H zhre_0t*)w>Oi0M2eIQOIiory16A@P~ML`4p0}_{t8yDhivI}8EL;`UmxDeck3vn@o zC@zGAfv6x7d?p!Wz#-17yQ*LHxW`3xPr9djf}?m=eXD=x_uX@<&Vm0qWbosicQ2d; z-uK=g+)HeXNw57bpZ>gNUdnQRXw>0u1lF_=&i(TFTNfDuG;ZC#saxBd+H5wo)k?L~ zX=}IJ(RQb;X`1Nj>L2>V%ZvWIAAi&JjSb!0-qJ?1sY#k@+G=T%rn=o|=&!d|^|3RH zEdVzFc<&F+%*^oDoi!9ic3sU_%j|5OwCiQ=!9#3q-odLM*BMLwXq_;yG6LvB5CI!F zjIoGFzXwX*%PEQifU+!66kBuHUAnB(+tG~yiXx*HMyM#(T0{&Yf~sOv$n&hP%L!`{ zY7}EE#ti8qL!M`->Np_Fa^g59FEgyIZifh}9>mejdVO6+9L40m(haIr^@s`R_PU_` zmJI_+UlK<###pSeSmy|wBd`Hs7?Jk@MOhHVQKc~!XDgkvjyRqo%X%miDo|BK46}#p zW1n8yN=DA=;m8DCr`sMmyWN82>sKcaj0ndbp5HlN|9*L|abbS$_&7jBk~B$X=8rwH zXJV2h#5K{i^+fKa+iG^maz6%m^U9Z3-c!|?y%I$veY7rLyn6nt7xQ9oowJfzBNt)9 zpdWkTLde7gaPL@ITIlxkF9Xj3&#x>kTy{go&qg)#Q7`v%`*l|^C(bNxW`)0`$}|1P zexyZHp}n;5%IV{$AHNSH`$MQIKdo#YdHaKF=XzD|KLfrV0knJNsi#ifj}^mR|6!00 z84jC!KshmYm|EbpmuH^OojAPT9)Awd)pL;9 zRk*tR4c6LyeMMzia&)$i4sI|BFcsP#mT&y|8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;JpsQj7}P}D;O9U@&kNATp5S}=}BtA0V+ZM${EROIqBNPX-ajO3I}TiFKlnRw5#>% zo{sDLdv0wly}hmM?$K%Yk4?LOeA=V^O^**WJw846@tJu~KxqEUi_8B1|L?o{wIc%q z1Aj@7UoZm`3oARHgru~Rs(S2#m5c9gsbXMYU`+CMcTw5G_H!l!1H%GO7sn8d;JJNo z`I-!PT(_U=^x4w!Iw0i3-ZkyKFN^;2cqTj6AG@!-_z~lz zTbt0^cV6|*0hMzuF4NX*ou+d!Qz+>D206}|e}8y2=y@e?Gd?z@wJmcR%hD@ZZknt? zH4k|!8zg_59#x#jz`&qd;u=wsl30>zm0XmXSdz+MWME{VYhbBsU>0I%VP$M;WoV{t zU|?lnFzvI2A&Q3F{FKbJN(LhXLqlByGhIXT5JMv?BXcW5V~7Svy>J0gJbAkMxvX004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm0000;0000;07l7cJ^%m!24YJ`L;(K){{a7>y{D4^00gN?L_t(Y$IX^`P}F4@ z$D!$t#%ao_Y^J#WVWgVO6swt{(UQvn3?V{FaJ*nBzyNhogn?Z~4v|aFmGJrKV~6&3tBO_ucpVeV_01{C>L{8vmC|+m!z_ zv5C+pH2*Eptwbu(OZ-ku6SKq|(M*I98~*8OLz$Sln-Kh@N=W3LxHdS>QB>0fesU$u z-IAH<6k-2�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 RR}cUI002ovPDHLkV1miz7Bv6> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/krfb.png b/Media/Themes/Umami/Icon/apps/krfb.png new file mode 100644 index 0000000000000000000000000000000000000000..67e018c0970054671f3860d41e6e2a7398381e92 GIT binary patch literal 1387 zcmV-x1(f=UP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00eDGL_t(Y$F-GhOqFFA z$A8arUceBPb&zw0!closTprNm)N%x40;SdxO-L(O>a1mc(JjrwQn&gNwi+O8nKC!$ zu%ygD2Ogu=Oqbbaf^VfD#{&YMm*IO1ZXiO^~B z`25(rU0lD_&84<>f@)~r;k|3NtbBgOyC##V44|{Kv+U@Rx?N)eO1Ia>hg7vODOP($cd7a3`@`KGan$b|XyxxK`F_7M@cZ=i4Z3V_oe?jhTnbGD_WIkU@gH~Qo9 zPbBSF01;GMa;<5})_K$EbM_JN`=Wj{H2ZQGgC1JReb&+Uatqq+edIo z5m)o_Ms{ER#NwPQHruIL0P@HK)ko+zZ|lNEX(T7aBFhqAHrU9rK8YlO5Fh|q zlJVX9gO-j?HQV4(QW z*Pw<3TCJYqr}J=h-Q&v5J6zwqo$SgA?2ax>H+<-I+So9_>XKFUYDityf3u$2PwS~2 zQV&&*HzrsXSXo>80++A;PJ2fuQ>LY&*U7ptfErSl?cVeLxW&ebw|DGh%j>Uk_S_|m z569rrYLI1(Q0>_GkY46gkaFsFo5mi z9V)lx0|FsH6#}=%j}Qa85D*;|fe^5{ylEJqC<;1_%(X6o*{L8&QV7tD0syq#^&`Xp z!0A@_{K%O>MIeB6r3Ktq`oq?^-ENwG_?d(mW?~|>Xkg3bRIpuy-ZG+ z%df4sP=o3xc6;X6yVi@L~0t^CS=PqR>Sb~wwH3+RaLcq zeWHGHRYi3apbTkYfLx1#T#I2sfWJp!Vc}kf!%>``mcD-DruCk|bXiqJ^}vm`?v%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 t05UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(bR}cUI002ovPDHLkV1o18hK>LL literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/kscreensaver.png b/Media/Themes/Umami/Icon/apps/kscreensaver.png new file mode 100644 index 0000000000000000000000000000000000000000..0543d365d00e941b596f815a7383ddb0391411a3 GIT binary patch literal 1116 zcmV-i1f%004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/locale.png b/Media/Themes/Umami/Icon/apps/locale.png new file mode 100644 index 0000000000000000000000000000000000000000..0bf99d0cefec6b3847c26c26a629db93ae9c543e GIT binary patch literal 1124 zcmV-q1e^PbP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00TiuL_t(Y$JLb0ZX87v zg}+8zMJ@RB6XRwX`5<-N9MMM?|1ql|2RUQDbB@&Oo3-AzLu=0`>5=5{-C{mE6 zEQ}=!j4Y7||6nF|dZyd`H$7d&VkY32N$d!-;7C`cDpl3F_nfN>|91$$M>lTI0B`XqtfU)QcKT`n~yC(4cKOETUV zC(RPC9R?GDA1ppT@ZbOaxBb}PZ+D|0F6qJuf)T`sh}K$HfEv&mNFai>aCNp4`3Z;H zK1cfzt*}k16ACG?JtdbG>XnzTO(F~zpzcuk0i4U%zXL%2~N@$I4aZukR zzVHy^#1yGZ5>@CNjgm)oOnMQQ>8HN9wOo=X#(#Ocli5%I-(yIn&G%P6W$xlNUTZX% z9Djogl}XYxBaGVwQH#UJP4*7g*^0g*ee1|Y%;Bw%OHWeNeG>VL;J8Uf3`0(Z>u)X* z$s-bbmtEPRs|>g$T&v8&=qt=mzlxy9Oh_hy%2i0m3dpU{y7%q+35f=1*)=)sVwWK` zjH6tVA|cLV;v^ql?97fOB3Ie7M!z{9*OsT>;$7AE>G8QZ3GnVCDYuonvQe=BDNJhQNzUBq!e3 z1McX`xO97$6=gL>)tX?~u9NG6mTq#8?~szdkVOB4F&M3J<0(QvrjhInqC-q5i{mWy-CnC9%z%<9Ovp zbvCb$O?dUuD@q(q$An~QNSFkt5=2IuPqtT!uh%{=Qo!rS5a0st*nFop^UlWWwfUcx z%5}M+EEy3Y14%_l1o3+^pAJQDGTQXIs?-mM-UQ1BL_u$&{`dG3>UC?jrz|1!0000b zbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3Hnt zbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(bR}cUI02y>eSaefw qW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000Sz8SAFmMeffFyxo-7dIRO9?3rV>Y0N5Mf*3AH*?|U)<0N9VSOR@n3o!g_d7XSc9Cs9fOE*%7DxeO2n zfEG7EBLXmA2|)Z2U{C!G$;rb2u&tg{@fkp(K0ZuN%Fh=8Fa_jdY99Pwu3l^Xch7%% zuD+p`dAa%Owbst>zV5wyr;mNNkHcbfSZpq`6$zU`ZjV7bXWHOu0W{cTku~@8T ztJP|?S#36(-TvHew@*(`I~)#&(=jtM<8(S*E|+W8H9I>yH#g^YyWJkQ$K#owpZ9vb z3kwTApU?02FD@=FEiElCFR!eu1OkCzFc=DjR##Wo*4Eb7*Tdm(Boc{6qp?_QV`HOQ z-5D3Qi$N(ZgH6~iTX(*+FXfH2;u7*n%J)5sHEWKzAIzDeg52{;43Y(a$tcXtrlve? z)p*L7$DxfTCAa&1Lv>tbkGV%H626JUVX2$iIkAmn;^PCYgiu5jI7(Ph?!YcTFSmF4 z+>g6dAGHNBpL9!l^Y6UU_(7NK-M_>^*z0UFd!r|4m}k3xD3VPQZpoRF{o7m=w0Y{b z@|%~Nx3^@`KCCFux>z$+`i|;KtQdVPl0?GlmBV#e9nf}p)5u48E%=I1kc4k1>LFn$KvGg@fzl|K&(ZTMiL`vv%Wn=Oc z{IRqm%^tG*NHBEd^r@tZ>y%wgz_>OiRyJzU=%IEdnRE*GOCk0G+`fTyB#9a@%*>=5 z#;dBYaKc}|w3qF^Nx~O5eeq&fv1-JHEtH)R@K#kn?cWI39>wH~E2ipcye0s6jgeQ* zI7?$x5$O$8G)5KR5j>KLyq$`qQ_)N!;VmMX{yKsX5#)Lx6a5cRTX*(c_2+;`P&5@u gPen7RD4vMVAfkkSfb6vQcE+z!m{*$1%c*4k4L_aFmH+?% literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/office-calendar.png b/Media/Themes/Umami/Icon/apps/office-calendar.png new file mode 100644 index 0000000000000000000000000000000000000000..fc012d70718ad3d2ec600feed780d388afa1ae27 GIT binary patch literal 915 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wDhK$4xH1q8`~rMK!-GS^gM&f>PnTwGjAN=kWoc|}DRa#K4Zp= zdGqGYpFe-`;>AmrELpmA>C)wkmn~bieEISfD^{#rxpMXD)w_1>+O>Pf!Gi}6A3l8i z`0-PxPMtY(=E8*wmo8npeEIU#t5>gGyLRKoja#>F-M)SM&Ye4V@7}$4@813U_a8iX z@bKZoM~@yoe*E~!lP6D~K7ID=+4JYmU%q_#>eZ{)uV24;^XBc_x9{G)d;k9ZhYue< ze*F09)2Gj$KY#i1@)aE`bfTh!YgV5nevDFsr;D~52;9+RkEjabzSr&$v{uL{P z8D2OXy~5nUq_s?*;Y1|6*57v}qMj}}=iV)FaB*^B3YpO1qQssoqgq;%L-qNS&M3ZeXH3czGPZrGuJos*PHHE?f>=Z&iT{z z^Z&0%sGd5ZJaqm2JN306JzKZ9gf5ctzayu&e&wOfrrk#smqyq3AOHBRQNx;{;b+>? zq}}}*f-`sTKQ6fH8!y8JTa%VcQ(YV`-`cf?W0JsyjJ1pm3=RKc8tNLD=^C1c7#dj_nOhkeLo_(*g$sb9z|+;wWt~$( F69ALPcg+9* literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/openterm.png b/Media/Themes/Umami/Icon/apps/openterm.png new file mode 100644 index 0000000000000000000000000000000000000000..b4d852c2d56c96836241a570f297291499862d71 GIT binary patch literal 1203 zcmV;k1WfyhP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Xs2L_t(Y$F-GNXkAql zhQD+7zEf^Zz=%boAc~c$IMEjoTZ%~4BBBo#Y8AwiJ~)7a;$uXLhy{`8lk`m~_z(y5 z#X8{B5I;k%T=A9R;o;B%^z~PV1_o|= zamzzntCebXbsD;McB+DEt5sB$rteq3{qE~sW0PYI;B^ZSNAaGAAKX$sF@B8k@rjl& zJTDXJftSr)s(<^2+t~iZlasfyJr)UxOD&)z$FO)#u!wDx85DhB`dmO0WB*s?@`c|^^f*`jxmNX2yo)a z2@;nQSZk@*7dZ6Gu{LncK>-9IGrnydoQs*8oyS^B5Cn+m3V~}87(=aA;nq6_a!_f@ zP*qSguNvQ9qT$ce^c&z}Oi{4b;0JzQ2&sZ3m1C@B{^BA>4o~Enrye_0K*@lE2Fv~u z;-pmr05q2(jIsEBGvA+#AkHi@l&d9%-rmQ--$wX*=2QwQsU?l(A`6W=rJ|#JO;6iN zN;{>pO%*iPQlN^lmZgPd&YhoO*E6s3?}f9xzwaHU&z&I%n`t6BU>82nT#m4|rJ#)< zj$@n?tT7ZzhL88ZhqXlppMQf-KRLj*9rseJ6oF!gCG7(bD3%Py8WQJndou+Iagwqx zE;QKr^dOIH-pSqBx2%)?#ddfg5h&*(-N=Kr`FKgR6)=Ty9nZ#D2lMw5CsX(?tH4#*_1^TBv@;4F6p$$ zNkSNfSYs)e0>$!**Js8q&>b7@zp+bE`YIR?&+~9jx-Ciw!w|3Ffx`Kzdj3Jrx-yp= zUcpNbp>v%CB0>_!%*>o+u_`sseU z$Pa?U$4CF%+g@Y+z_mbcM(VM;I4`*c*^>Z^KpmKCfs}w^8#q@<&}tJP2BIs(xbpZ9 z1fW<*2_qG{0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckW zFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(b RR}cUI002ovPDHLkV1miz7Bv6> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/preferences-desktop-accessibility.png b/Media/Themes/Umami/Icon/apps/preferences-desktop-accessibility.png new file mode 100644 index 0000000000000000000000000000000000000000..29cd13746bbf9c46313920008e7cc6c72a060ca4 GIT binary patch literal 1171 zcmV;E1Z?|>P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WgtL_t(Y$F-GPXk1kk zhQED1Gt=COHnv_$#G+P<6ui7>f{GN1ZG%{-1fLWycu6Z(h58bzV3mRoR$Ct|LL-FK zw>}6G4N~+$G{uq@Okzv5BynL_n6=H z`qBL*TwR*vop|-ry<0hZae>c1K1D7cQV3(LQ8j1K)p0xTzVtZHzSI9Ca9|lhHkaOT zdyjJ-=Nz$f3=EF&#KU)TVZMHIMFva5c`!z! zl`4`5_U?Xw_186UdUTTGU!4JbnILu!7drw|30IZC7>(AlOmK5sBOP0kx;NAX419YI zJh-J9#ICAr6$ApXMxnJT6XfzSVdOAc5=JqBHF#f=C;^Y8ThWtA$chyqFh(cCYOK{1 zV$W+I4U)@;^d9@3z#0OpNd*RL4c2O`(NzhmOQaRr=;Wys>A(;>kI{;QFLV+ZMLvuf z9-ZQgug?-oK^zyV;a8VPYlSI}=7uzHb>C0#=RZ)0Vm|2qiCiHn%VVu3iUeU8(^zN! zn;?_2EXK)cjnTZg^Bx9R)+mhC z1S!q#&Rh9$G|QF6h}MQQo7-DaQX;O>MdeI{Ah84?kU}s#mc?4l%iHh5<4IY?cO!rD z>!mAbCHds!2!%qJSjqA}(m+%KPF3qBk+5-ke42B=&tr{32#F00Qpn;#iXS6Z-h&jd zAS(z2WUMTk8Qzy~tp}tML{XIZk5Bx_dytj#q?Sq3Y%Wh+oSbWW?A~^U$Fc}PG3^*4 zM;I4~784i6dz=UFJRXm7o+ygg_Rww2&d#tdm>OTnR=2Zj`>w4$$N%>ar)yuI@9cj2 zAW&D5x|B&%$5Z>?FpX_b&qYnwvx>OEPM)8iogEnJJMbPb2SiJi(?!-4kXdm@^uJ(6 zE9QR#i@-u@uO@%0k>F};*Dn46oWt2%*+|cc0000bbVXQnWMOn=I%9HWVRU5xGB7eQ zEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q lEipG#F)}(bH##vhD=;uRFfc(bR}cUI002ovPDHLkV1n(6{k8xA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/preferences-desktop-assistive-technology.png b/Media/Themes/Umami/Icon/apps/preferences-desktop-assistive-technology.png new file mode 100644 index 0000000000000000000000000000000000000000..5f137b3392ae6ee0f9ca88196aaf7ff83d6fcd18 GIT binary patch literal 1233 zcmaKsYgCeV0EhpYr8C_;J2vgIowZWzqP^I%Vl^+}guGu$%TO#3!H^e3R5C@xai31| znjTC+OSzbeib|5V$dkvFH7na}?1H+9b~9&n+VSkuzU+6-^YQuioS!f%BE;Eoiz5KQ z8H+*3007+Aa99Tbn7TGT1^}=l1%?L#aJ^*X+m!VHfX~liV!{Dndji1W0k8@H;LHM$ z1pzRR2LM$CfE!%f8+`--@BxfDl?j0J)EYKHUYBzKu=R=xiwg$-yQt`hfFS>)UmuBt zYa;2@F^uYySvCHr^I|hAF&USGDdkZ_Hky3tOQPZ^UK~KG3nI&dDDq zTTZZ5C-cN5ozT!9A1ZBNRnN-Ank zD(Xn)Xp;-I$y{x6QAaYb1JBjsi#t=k)t%*c5sKRhf-Zufn<(riih5E-JtTf7Nzh3W z=t#mYl2}iY=+n6MX+m9^NS7w8F%4=~D&jEb9?vi{7=Z+sGx4BP(I9+k8l-timr?nHI5Z2$9YZT#SORkB0gWt=Zgh= ziC7>i7DxnwYZD@=KvE)+mP(|RwUyPf%bk-AjrA(S1J#Wm)q_to15+KQr#-!$h5^0N zc*8h8W1KJ#j*Jf9GL76d-x?d5oSK+6FFds@nP(S&owfcp|IA{2HD`Tkwf<$bE-b!y zz4T&b>E+6+mDS~y)z>R;R^PmP^Y-2Q_wRp>9a$^IjTsXW3+x=6TwLA1@Z5cpK!)jA zdE&b>7Ry}1)Ot4nfX&&^5VUlwllcJvHicMpV4T;)IlOV*3U4Hu1NozC<4HfUT-$tN zz~1fsjO*I+E1PV?k`Z-cWvqYCN(9oTl*YlwfXIbi$fpIZB0bwA~HU7baYr^ zwZq3SRS$FiMhsJ@+lWJ4*grk8hiA~m2mG>nC;Wl_k-=cF-iC$v5>C(Wv0}I7WuJN6 z^q&u>H_9BI39*k8k|z6DNsFDzNw!73P>i)3C8=ptYhAai?H@OH2JI+&o~p5G=W(d! z3u8Qe=mn1ev&1~zcs5o=Pr#c-`^El$@Naz zH1k>Ay8CxG+(VKX`KJ(Z_N^w-saMX@aPLn&4C$7z+Iun84}{Bih)h`DTsI$?TUrW@*9t|OHV zMbFC@J8sA(u3dUNDkJrdXxf#hCt2rcaiFK60dSWBFjN4q)btpNjsrks4g1VRfGA5im~6885&)*7@RT42{#S~2xuRX(-f^w7 zOIcWyU&Jrq6`WGE%Ns69FV@un00x7>VzD?J4v)u2M@PrS#l^?RCnO{!CMJ@}O--#Wt%`QJN~Kb()jd5u8ja?bR@2wlH!v`u)9D5W2fx#QtJmv?hK7cRhet+6 z3c5ZHN{^7%gg@wh%#l@wt z_hnzQFD=>aPhnrSFYhBP$MUknv5)X|IGs-CKAf(wT`rf~<#xH;ZuiQ{%8JM1*@xHb zU0q#WTU%RSU-$WZe!o8u2yAR@Y;JCDZEXdE!R_tsP$;ysv$MOqySKO3UwiTk05~M& z=f4Noi06ohV&fB%($X2I@bsD6hOyl&zdHQ$@;nwwXN@Xm05I=!P^KXEFz%;muAnRy z{&rmXJ!9flOp@Z))|kPzej#%=qFEvQ>tyhcl<%vpbfpdZ5~aHt&(*$sl|;W8x_9#+ zKvwBo0cPw;C{q65Q*$J!yjPA3i*Fox)!r|b=+54CxOMM#yM7_(op9?#Z{EJ&b%Azg zj34xRM!dSInwo(h2$5D@LABEE^ZRF8ECCE*!*22P)xLH|bN0xSy24(QRTG*KRK^^; zi+h3V3z;?x)@W%9k886(@AxXl=T9E${JB%zn{|CspYtB~d)*1=%-Bvlcfa#$rgi2RSbi ziM`AN0J6Qq`pT(#*0Jz(-jL=LODoExV#c`(;xefiXb25SLEcG0=mKgcBkgSlmHrlj zFc9Q=Ad~u^!pEOfR7o0vhES;~2t9?GA)wM2v3ip literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/preferences-desktop-keyboard-shortcuts.png b/Media/Themes/Umami/Icon/apps/preferences-desktop-keyboard-shortcuts.png new file mode 100644 index 0000000000000000000000000000000000000000..017ef9bbb7ad331c276d8804e6ee5a488b5edcd5 GIT binary patch literal 1163 zcmV;61a$j}P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 N002ovPDHLkV1g!+#ufko literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/preferences-desktop-remote-desktop.png b/Media/Themes/Umami/Icon/apps/preferences-desktop-remote-desktop.png new file mode 100644 index 0000000000000000000000000000000000000000..67e018c0970054671f3860d41e6e2a7398381e92 GIT binary patch literal 1387 zcmV-x1(f=UP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00eDGL_t(Y$F-GhOqFFA z$A8arUceBPb&zw0!closTprNm)N%x40;SdxO-L(O>a1mc(JjrwQn&gNwi+O8nKC!$ zu%ygD2Ogu=Oqbbaf^VfD#{&YMm*IO1ZXiO^~B z`25(rU0lD_&84<>f@)~r;k|3NtbBgOyC##V44|{Kv+U@Rx?N)eO1Ia>hg7vODOP($cd7a3`@`KGan$b|XyxxK`F_7M@cZ=i4Z3V_oe?jhTnbGD_WIkU@gH~Qo9 zPbBSF01;GMa;<5})_K$EbM_JN`=Wj{H2ZQGgC1JReb&+Uatqq+edIo z5m)o_Ms{ER#NwPQHruIL0P@HK)ko+zZ|lNEX(T7aBFhqAHrU9rK8YlO5Fh|q zlJVX9gO-j?HQV4(QW z*Pw<3TCJYqr}J=h-Q&v5J6zwqo$SgA?2ax>H+<-I+So9_>XKFUYDityf3u$2PwS~2 zQV&&*HzrsXSXo>80++A;PJ2fuQ>LY&*U7ptfErSl?cVeLxW&ebw|DGh%j>Uk_S_|m z569rrYLI1(Q0>_GkY46gkaFsFo5mi z9V)lx0|FsH6#}=%j}Qa85D*;|fe^5{ylEJqC<;1_%(X6o*{L8&QV7tD0syq#^&`Xp z!0A@_{K%O>MIeB6r3Ktq`oq?^-ENwG_?d(mW?~|>Xkg3bRIpuy-ZG+ z%df4sP=o3xc6;X6yVi@L~0t^CS=PqR>Sb~wwH3+RaLcq zeWHGHRYi3apbTkYfLx1#T#I2sfWJp!Vc}kf!%>``mcD-DruCk|bXiqJ^}vm`?v%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 t05UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(bR}cUI002ovPDHLkV1o18hK>LL literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/preferences-desktop-screensaver.png b/Media/Themes/Umami/Icon/apps/preferences-desktop-screensaver.png new file mode 100644 index 0000000000000000000000000000000000000000..0543d365d00e941b596f815a7383ddb0391411a3 GIT binary patch literal 1116 zcmV-i1f%004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 FV1nvdCe#1` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/preferences-system-windows.png b/Media/Themes/Umami/Icon/apps/preferences-system-windows.png new file mode 100644 index 0000000000000000000000000000000000000000..2a02a2e16c9052ee769755512ecc4154a6191bd2 GIT binary patch literal 714 zcmY+5do0ue0LQ=La9*8kxq03^PR-7KcR2i#9O~|tJC9CW3CCR?8;aARX-7EzqmIn0Y3i@`+2_B{rzTd+x3zMz0$?i?M8yN3?`UZb zU|Mx>4S-oDPs9VB)mv}LC;%84f_M>7$^_Jvzy<*IEKo!Mb7=tC063m$?2ZcsV4i-I zABF85nqRbN08q~eq=kUB!;U7n*+>UW?g*oJ62aht=7rcfj&@{Zo~#lbt^S~o3ulyP zjaI2Hd0g2SLwg@7GRarVy*yjKQ;ip^n)!F@(t~t11B<6pJ5&o4U)R*{okdHHH$qa1 zhbjccbem0gwI=iH)WnxhA-P2b#QbKVKYg_#$vMD%O-JUb@hplD-;CkY%Kas$~O7IA2VW-_W1FV`HFa4mY<+R~ivKJN8qOZK;-GY0JV8QEpr&gxD&vHPTae1)4y;sXT z8)K*qzQvF(3VYW^2Y-7`KTl4ylZSKZtNi`zVWST`;~&52*>E(d&DFcM)n}ziZ%4X} zwdF3=&iM>_?B@ z;CDXSaBQQ-p$Gi8RyagxGE1PDI@c9a}{|?L>4GyN`;)E%2sAE zBv}duUnq)Ql6&U?@ZJgKGNppd_*;QPa0nI=!Xmg5b};E5NV2)U1VIwS_2pppe?xA* oJo|JBI0T!`BDgGekc7=4IYA`b|KGs#OQZpS@US@QNkm%3ANV2_W&i*H literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/proxy-config.png b/Media/Themes/Umami/Icon/apps/proxy-config.png new file mode 100644 index 0000000000000000000000000000000000000000..f30dca7d4d2b96d8fbda3e0e567ccc37e6b9e028 GIT binary patch literal 1358 zcmV-U1+n^xP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/redhat-filemanager.png b/Media/Themes/Umami/Icon/apps/redhat-filemanager.png new file mode 100644 index 0000000000000000000000000000000000000000..3597ca1e1642608a57b12124375aa8640f659374 GIT binary patch literal 559 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;JpsQj7}P}D;O9U@&kNATp5S}=}BtA0V+ZM${EROIqBNPX-ajO3I}TiFKlnRw5#>% zo{sDLdv0wly}hmM?$K%Yk4?LOeA=V^O^**WJw846@tJu~KxqEUi_8B1|L?o{wIc%q z1Aj@7UoZm`3oARHgru~Rs(S2#m5c9gsbXMYU`+CMcTw5G_H!l!1H%GO7sn8d;JJNo z`I-!PT(_U=^x4w!Iw0i3-ZkyKFN^;2cqTj6AG@!-_z~lz zTbt0^cV6|*0hMzuF4NX*ou+d!Qz+>D206}|e}8y2=y@e?Gd?z@wJmcR%hD@ZZknt? zH4k|!8zg_59#x#jz`&qd;u=wsl30>zm0XmXSdz+MWME{VYhbBsU>0I%VP$M;WoV{t zU|?lnFzvI2A&Q3F{FKbJN(LhXLqlByGhIXT5JMv?BXcW5V~7Svy>J0gJbAkMxvX004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00eVML_t(Y$K94|Oj~6b z$7j^UOH7Q3lBfxOm@%2g#F&_UF`9W<7B2yu1EzRO+!Eb%7c2vrXrtg55;wN8ZjAeF z1zTF6&y zSJ46XIhp+aZ=;S$N-MMEmaAcQOh<$*%m_lE5F}DLR_=?LSwW+*<(Ss!2+>T3%%V>qHf5HE;vTVVK6i385l!eRcFLxTk#}L zDgR8F8qUcX^bR->4u{cWvcVM$!#T4GW@;H4%Q)J5%;?aWFu$;f`T2S1h9;18S+RAx z*n+V+Wm6IgtwqFJ+d5!%2jQ9r+{B3QCQK98;hFYgd3iaqW@TjsQ&S#DdK^euugT_x z|L|B5Kfk&YbBpU6g)J(K&jAy`e{lFi(DpMBG%8WA=tI3!4WEAoUauGZ!%lFowy_4> zO$|OeRpt?QSn-$F4`Gu6rpcQaBLkN4C^qjZ4AIn0(>k@zf=0CwKA#T*LkzA+jI6=t zwMfaV40R8!$2voyZT@7t8n4O(PufYgTb|Gro7$l#n zSlGNpzbp0CDhH7(WMCh4t#1y@p`^ux@{UOqt6V6QjpA%YJGjM4boUKnad8oDw;TQa zCL|o^fhRJsZ=l=MAmPX*QBJ7}yk=%Yglx+Sq+R$;Y(q{xgCDBRNWE$RuT}w%#{;X? zio6TOIGQiTX};q2T@-hy;AOUtr8+0%8hvCm+%lN|I@nY z8=MBbexeS0Q)_Yfj0EMfL2xMY_@fuSF|oUnX#WeugIgL9FA?wV`z+g@l37hdOo>8K zAF7lzBGemK{u?wmb|-w($h zxwxF1R<=eh2%5NSabFi)e=XtUAVqza)_p;|6AkR4J3V(EC3X?767R9cL{#W7aVQ#b zBC(Hng?OB}=Wb5U`-q2$$A~A1SmLRu;3LHS|M@?MzX7|+vv@@3Z=?VK03~!qSaf7z zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6W zZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*7SLFgh?WK`vJi00000NkvXXu0mjfQqg1P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/screensaver.png b/Media/Themes/Umami/Icon/apps/screensaver.png new file mode 100644 index 0000000000000000000000000000000000000000..0543d365d00e941b596f815a7383ddb0391411a3 GIT binary patch literal 1116 zcmV-i1f%004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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�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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/system-users.png b/Media/Themes/Umami/Icon/apps/system-users.png new file mode 100644 index 0000000000000000000000000000000000000000..4e4b59b80c778e5a3056f6c3a7825c054d819099 GIT binary patch literal 1402 zcmV-=1%>*FP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00ewVL_t(Y$HkNjP*hbM z#vusz@`zBgX&46zr&LA>SRM}Iz$l6m$Wz^QcY$T)we5qq(eQpM7GgC;;CuR=j0VMIb~2;%G9WpOrwVzj9<^Ub-t`#<0B{J(SVs;T`S!{1q(#doH9 z$UIF;`ONxP%=ymRpT_Q`erc7goVj^eAGBJ;FV7@&-dXPV>Q#J4#P=*`=W_K&yYe2OJ%!VF0# z`ZJ`am#%G%GoYR&`)N9o3&gM|*BL|A9Q2)j2hB%pQ5Io>M(L|4EuI0O+U#Xzbqu4;F&Jrz z#Bd!SLo!!9Y!cy4s_A4mNA2%#n%^#pljuqPCXdpEOl*jf@HbE>F+zFBD*PaR5mQ$a zF?uNyV{OOrs5=*!w zt56-f8a+I9JZaCwgGMpN+H)}8lQ##b@zL;4h@W7tP7{z?^wjX&d%<9K-t( z37SGTpwm|i?K~~WxF!&|hG4f_8vHUEp@gLUwS!F8SrK_#5AItBP9Ou^AX++#1s{}s z2kX$YuoZoQnuZR@%3I*?8jnPN1}cggQF`_pSOu5ELU86ORwA{5 zD9lVDq0}54JC2ht)wN#3<&NuMCDjr0Vps&6QAh=#qYBU@(j!_! zw3f(-$b`t8F6O1wDgfJjJ|uIYLc**%_@`9O8S_uW-YbjD1*(A63(s0z*%l4cBBK9} zU!#lnCDy|`p&ok$pJC78S~!GN&ndk6>9ezME*!~cr|8-x0gC~3Tui<5ls9A9a@(~3 zqlFPIH8hlkuc$;v>-7bi|9-8qR%hWM|5GRP(P^qgs4CUpG?Z)poACz%D%R+JJ}}n+ z001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Xs2L_t(Y$F-GNXkAql zhQD+7zEf^Zz=%boAc~c$IMEjoTZ%~4BBBo#Y8AwiJ~)7a;$uXLhy{`8lk`m~_z(y5 z#X8{B5I;k%T=A9R;o;B%^z~PV1_o|= zamzzntCebXbsD;McB+DEt5sB$rteq3{qE~sW0PYI;B^ZSNAaGAAKX$sF@B8k@rjl& zJTDXJftSr)s(<^2+t~iZlasfyJr)UxOD&)z$FO)#u!wDx85DhB`dmO0WB*s?@`c|^^f*`jxmNX2yo)a z2@;nQSZk@*7dZ6Gu{LncK>-9IGrnydoQs*8oyS^B5Cn+m3V~}87(=aA;nq6_a!_f@ zP*qSguNvQ9qT$ce^c&z}Oi{4b;0JzQ2&sZ3m1C@B{^BA>4o~Enrye_0K*@lE2Fv~u z;-pmr05q2(jIsEBGvA+#AkHi@l&d9%-rmQ--$wX*=2QwQsU?l(A`6W=rJ|#JO;6iN zN;{>pO%*iPQlN^lmZgPd&YhoO*E6s3?}f9xzwaHU&z&I%n`t6BU>82nT#m4|rJ#)< zj$@n?tT7ZzhL88ZhqXlppMQf-KRLj*9rseJ6oF!gCG7(bD3%Py8WQJndou+Iagwqx zE;QKr^dOIH-pSqBx2%)?#ddfg5h&*(-N=Kr`FKgR6)=Ty9nZ#D2lMw5CsX(?tH4#*_1^TBv@;4F6p$$ zNkSNfSYs)e0>$!**Js8q&>b7@zp+bE`YIR?&+~9jx-Ciw!w|3Ffx`Kzdj3Jrx-yp= zUcpNbp>v%CB0>_!%*>o+u_`sseU z$Pa?U$4CF%+g@Y+z_mbcM(VM;I4`*c*^>Z^KpmKCfs}w^8#q@<&}tJP2BIs(xbpZ9 z1fW<*2_qG{0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckW zFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(b RR}cUI002ovPDHLkV1miz7Bv6> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/text-editor.png b/Media/Themes/Umami/Icon/apps/text-editor.png new file mode 100644 index 0000000000000000000000000000000000000000..7ed1d9916ba77df7b761d6aed37575395a693140 GIT binary patch literal 1007 zcmb7@dobGv9KgRpEf`DBx;kTHT`eBTexZ#t=jPEwrS09^W+Q;f9k z&dARIaEDloP?UJK=*yXN!y$xl&ePM<^v8o7DKY%uG0D=oShOM|tXwR5^{HKO{#V8* zGA$=C>u-{hs7$S_+8K#p%{LUWz9n*+f_sM(3l61J8Xhw3J#2~sj8S8qcet&M^j3_QRNnHznIT_-(t1>8zOr7tx`kT%Vk6xY-d|fj7 zUhPJ3jZTcG%=OJ#_qoPkzghC$zA6l?xkx6lH%6AZMZPs5wRL5uUu<7J@}^JBF+odO zp4xPy_q?i|z`D1jC8xh;T6Z%w|HKd9`E^cjXLx^m{^6F=wItIysM2SGD5};l&Mdyv ze&jjv$4#WYnkYHnEYWy?O`#(2^`2AadqS?uYc`olr@$TK^4tK-=Z zu3_G<1QMnnH8I&X0|1U<%w#Mh4dc=|*STpJ7yJ=_goKeZOLPwCA xO&aAt0Z))|DYF3l5ekKbP)U?Pl;ThK52RB9{tZIONre5+Sj=d~cVVfe{{Tvgs0{!B literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/update-manager.png b/Media/Themes/Umami/Icon/apps/update-manager.png new file mode 100644 index 0000000000000000000000000000000000000000..bf5abd6137349a72e133e088120642c0eb28b374 GIT binary patch literal 1518 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00i(!L_t(Y$Gw(qOjB1F z$D8c?{IYDBS+ZquV_C8#Q)7%-GM!^?$~NUiH;`$fvV?5P9Kp%z1m<8Ob52M*pzD{LBzd->o^e#t%e-t+&R^W1Y!E{F4^ zj!hqY8$n<-VpA*BpPek9CIW=*3~csM=zGmj_uWL!I0nB)a-+2m}I98?9h`0gOy8!aOk#m1z`h>Rz;KjhMbOgX!sMXa~k{ zx}j+?su$hQDF2oFy{Lko5v4Lj~UgCGF+2(9V7FwxC5|z0Xv1; zCYR7-^ME~A!3xZ{6Ut5q1SJ3$6V=sBE+7W){#l zu>b=zhYo|~p>USj*;!Q7wn8RXpsr(>T}=TJ;1NMwYQ|#qZmN5 z$&035FN6jUsuhD+T=YLAn3|eGm)?MsOg^gQW^f~u=0XXoK1#^-U6J)8yVi_hJ9EFY z_cn@Kj3{d#N0EX7Uu?&@@-}1?$Y zgAFHCAwRaLqXliJX%t1RW^@@$@Or($FZD-$4H9{s!QcZq3hd34ASvf6G#U+B#bTVl zti*}D7Mu-_Z(J6nhukjKzk4il-#0~fekxO;wyhV|p&$cE1?5IcvlA1NM#K2k*t~ zi$#H;1Tp*Kup#`wABSGw@?c4@YI$N}V%8Di#8Kh`A&iQOvPMTo-=Xs*0xS%}1X$?B z2H5s0;X$g_u08sfK&#cRA;O4Hh-9LWP!n!qv81E~X=!P!0MrKCPDn@yUZe8>U7I7U zL@9BQSWmq0;0)FfVZ8;fC;$Ke literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/utilities-system-monitor.png b/Media/Themes/Umami/Icon/apps/utilities-system-monitor.png new file mode 100644 index 0000000000000000000000000000000000000000..84a05628a6c814140ad983bc7c682f10eeea4ecc GIT binary patch literal 1160 zcmV;31b6$1P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 RR}cUI002ovPDHLkV1miz7Bv6> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/wallpaper.png b/Media/Themes/Umami/Icon/apps/wallpaper.png new file mode 100644 index 0000000000000000000000000000000000000000..fe80dcee3060418d8fc581705cefc75733b16e85 GIT binary patch literal 1170 zcmV;D1a13?P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WdsL_t(Y$JLZgXk1kk z$A5PwnR&@`CQW9{Xw8^3LSsyAZ8a_gwSpyxU_eSiK_Q@47p^KH;!05vckYY}H`ax= z(hioo>Bf&VZ8bKLG&a+;O)`~srkzaZ^L@YTVqQXIrkPOOc;N8fJNKURyZ>`|7yj=i zR$o5-?31^E@&Bmq!LiW~z65Am7aSiR9!61G4bwE8+8l|LfRsoi(glg@#Xw4l?buRL z0%{_W7zb(q037-DXfTt>0B|N*BCiQ<<_)Bjb&ncrYhvH7F!6YtXlFZ-NaSDZHGoVe z!^p@8uIo}ca)v;&pOe2PDQiYuK-Prz2lsF-b)KOYo(EuNX6DWS0A1ISQnED`;qmQJ z=2AKm!KGBLE@WaZ$3O_Crlx8X`06E?rip2qs6ihs0f@E*80_yv2tbG$znCf@CG6h4 zhn}9Ejavj@Sr$@Cgpf$r;r`}*zH^9Mxgv{K7ILSsNWo%UuRP*s&ss4BEK zw!;3g11K$9Sa%5k;JPkCNG2yvQEmhX27~x~)uKX(wPy+F+wl}lEj=|k($xW&oH(_y zg08MErl+Ut%I}VVS(w8qFVq64|EJdgUVLe&dM(Iml9Y1oJAM5s-yi#KZSUYf5X18E zXHr{>ZR;CkcA-=k@I@k#c$-G%?*~A2{-T_}PDumkpzOtG7fRT+{nhGrafdQ3KzMZQ zy~$^uepbD@csXhqhR-z3YC9x^nP>130X~J_W;Nb@|8O^u_k80a1I<7%oz6@Yi-lMA z?tRX;va&*1dCLWOCjF0004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00eVML_t(Y$K94|Oj~6b z$7j^UOH7Q3lBfxOm@%2g#F&_UF`9W<7B2yu1EzRO+!Eb%7c2vrXrtg55;wN8ZjAeF z1zTF6&y zSJ46XIhp+aZ=;S$N-MMEmaAcQOh<$*%m_lE5F}DLR_=?LSwW+*<(Ss!2+>T3%%V>qHf5HE;vTVVK6i385l!eRcFLxTk#}L zDgR8F8qUcX^bR->4u{cWvcVM$!#T4GW@;H4%Q)J5%;?aWFu$;f`T2S1h9;18S+RAx z*n+V+Wm6IgtwqFJ+d5!%2jQ9r+{B3QCQK98;hFYgd3iaqW@TjsQ&S#DdK^euugT_x z|L|B5Kfk&YbBpU6g)J(K&jAy`e{lFi(DpMBG%8WA=tI3!4WEAoUauGZ!%lFowy_4> zO$|OeRpt?QSn-$F4`Gu6rpcQaBLkN4C^qjZ4AIn0(>k@zf=0CwKA#T*LkzA+jI6=t zwMfaV40R8!$2voyZT@7t8n4O(PufYgTb|Gro7$l#n zSlGNpzbp0CDhH7(WMCh4t#1y@p`^ux@{UOqt6V6QjpA%YJGjM4boUKnad8oDw;TQa zCL|o^fhRJsZ=l=MAmPX*QBJ7}yk=%Yglx+Sq+R$;Y(q{xgCDBRNWE$RuT}w%#{;X? zio6TOIGQiTX};q2T@-hy;AOUtr8+0%8hvCm+%lN|I@nY z8=MBbexeS0Q)_Yfj0EMfL2xMY_@fuSF|oUnX#WeugIgL9FA?wV`z+g@l37hdOo>8K zAF7lzBGemK{u?wmb|-w($h zxwxF1R<=eh2%5NSabFi)e=XtUAVqza)_p;|6AkR4J3V(EC3X?767R9cL{#W7aVQ#b zBC(Hng?OB}=Wb5U`-q2$$A~A1SmLRu;3LHS|M@?MzX7|+vv@@3Z=?VK03~!qSaf7z zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6W zZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*7SLFgh?WK`vJi00000NkvXXu0mjfQqg1P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/xfcalendar.png b/Media/Themes/Umami/Icon/apps/xfcalendar.png new file mode 100644 index 0000000000000000000000000000000000000000..fc012d70718ad3d2ec600feed780d388afa1ae27 GIT binary patch literal 915 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wDhK$4xH1q8`~rMK!-GS^gM&f>PnTwGjAN=kWoc|}DRa#K4Zp= zdGqGYpFe-`;>AmrELpmA>C)wkmn~bieEISfD^{#rxpMXD)w_1>+O>Pf!Gi}6A3l8i z`0-PxPMtY(=E8*wmo8npeEIU#t5>gGyLRKoja#>F-M)SM&Ye4V@7}$4@813U_a8iX z@bKZoM~@yoe*E~!lP6D~K7ID=+4JYmU%q_#>eZ{)uV24;^XBc_x9{G)d;k9ZhYue< ze*F09)2Gj$KY#i1@)aE`bfTh!YgV5nevDFsr;D~52;9+RkEjabzSr&$v{uL{P z8D2OXy~5nUq_s?*;Y1|6*57v}qMj}}=iV)FaB*^B3YpO1qQssoqgq;%L-qNS&M3ZeXH3czGPZrGuJos*PHHE?f>=Z&iT{z z^Z&0%sGd5ZJaqm2JN306JzKZ9gf5ctzayu&e&wOfrrk#smqyq3AOHBRQNx;{;b+>? zq}}}*f-`sTKQ6fH8!y8JTa%VcQ(YV`-`cf?W0JsyjJ1pm3=RKc8tNLD=^C1c7#dj_nOhkeLo_(*g$sb9z|+;wWt~$( F69ALPcg+9* literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/xfce-edit.png b/Media/Themes/Umami/Icon/apps/xfce-edit.png new file mode 100644 index 0000000000000000000000000000000000000000..7ed1d9916ba77df7b761d6aed37575395a693140 GIT binary patch literal 1007 zcmb7@dobGv9KgRpEf`DBx;kTHT`eBTexZ#t=jPEwrS09^W+Q;f9k z&dARIaEDloP?UJK=*yXN!y$xl&ePM<^v8o7DKY%uG0D=oShOM|tXwR5^{HKO{#V8* zGA$=C>u-{hs7$S_+8K#p%{LUWz9n*+f_sM(3l61J8Xhw3J#2~sj8S8qcet&M^j3_QRNnHznIT_-(t1>8zOr7tx`kT%Vk6xY-d|fj7 zUhPJ3jZTcG%=OJ#_qoPkzghC$zA6l?xkx6lH%6AZMZPs5wRL5uUu<7J@}^JBF+odO zp4xPy_q?i|z`D1jC8xh;T6Z%w|HKd9`E^cjXLx^m{^6F=wItIysM2SGD5};l&Mdyv ze&jjv$4#WYnkYHnEYWy?O`#(2^`2AadqS?uYc`olr@$TK^4tK-=Z zu3_G<1QMnnH8I&X0|1U<%w#Mh4dc=|*STpJ7yJ=_goKeZOLPwCA xO&aAt0Z))|DYF3l5ekKbP)U?Pl;ThK52RB9{tZIONre5+Sj=d~cVVfe{{Tvgs0{!B literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/xfce-filemanager.png b/Media/Themes/Umami/Icon/apps/xfce-filemanager.png new file mode 100644 index 0000000000000000000000000000000000000000..3597ca1e1642608a57b12124375aa8640f659374 GIT binary patch literal 559 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;JpsQj7}P}D;O9U@&kNATp5S}=}BtA0V+ZM${EROIqBNPX-ajO3I}TiFKlnRw5#>% zo{sDLdv0wly}hmM?$K%Yk4?LOeA=V^O^**WJw846@tJu~KxqEUi_8B1|L?o{wIc%q z1Aj@7UoZm`3oARHgru~Rs(S2#m5c9gsbXMYU`+CMcTw5G_H!l!1H%GO7sn8d;JJNo z`I-!PT(_U=^x4w!Iw0i3-ZkyKFN^;2cqTj6AG@!-_z~lz zTbt0^cV6|*0hMzuF4NX*ou+d!Qz+>D206}|e}8y2=y@e?Gd?z@wJmcR%hD@ZZknt? zH4k|!8zg_59#x#jz`&qd;u=wsl30>zm0XmXSdz+MWME{VYhbBsU>0I%VP$M;WoV{t zU|?lnFzvI2A&Q3F{FKbJN(LhXLqlByGhIXT5JMv?BXcW5V~7Svy>J0gJbAkMxvXSz8SAFmMeffFyxo-7dIRO9?3rV>Y0N5Mf*3AH*?|U)<0N9VSOR@n3o!g_d7XSc9Cs9fOE*%7DxeO2n zfEG7EBLXmA2|)Z2U{C!G$;rb2u&tg{@fkp(K0ZuN%Fh=8Fa_jdY99Pwu3l^Xch7%% zuD+p`dAa%Owbst>zV5wyr;mNNkHcbfSZpq`6$zU`ZjV7bXWHOu0W{cTku~@8T ztJP|?S#36(-TvHew@*(`I~)#&(=jtM<8(S*E|+W8H9I>yH#g^YyWJkQ$K#owpZ9vb z3kwTApU?02FD@=FEiElCFR!eu1OkCzFc=DjR##Wo*4Eb7*Tdm(Boc{6qp?_QV`HOQ z-5D3Qi$N(ZgH6~iTX(*+FXfH2;u7*n%J)5sHEWKzAIzDeg52{;43Y(a$tcXtrlve? z)p*L7$DxfTCAa&1Lv>tbkGV%H626JUVX2$iIkAmn;^PCYgiu5jI7(Ph?!YcTFSmF4 z+>g6dAGHNBpL9!l^Y6UU_(7NK-M_>^*z0UFd!r|4m}k3xD3VPQZpoRF{o7m=w0Y{b z@|%~Nx3^@`KCCFux>z$+`i|;KtQdVPl0?GlmBV#e9nf}p)5u48E%=I1kc4k1>LFn$KvGg@fzl|K&(ZTMiL`vv%Wn=Oc z{IRqm%^tG*NHBEd^r@tZ>y%wgz_>OiRyJzU=%IEdnRE*GOCk0G+`fTyB#9a@%*>=5 z#;dBYaKc}|w3qF^Nx~O5eeq&fv1-JHEtH)R@K#kn?cWI39>wH~E2ipcye0s6jgeQ* zI7?$x5$O$8G)5KR5j>KLyq$`qQ_)N!;VmMX{yKsX5#)Lx6a5cRTX*(c_2+;`P&5@u gPen7RD4vMVAfkkSfb6vQcE+z!m{*$1%c*4k4L_aFmH+?% literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/xfce-man.png b/Media/Themes/Umami/Icon/apps/xfce-man.png new file mode 100644 index 0000000000000000000000000000000000000000..45c3c9318833af07f9fc5e8c651365ccbe6c849f GIT binary patch literal 1484 zcmV;-1vC1IP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm0000;0000;07l7cJ^%m!24YJ`L;(K){{a7>y{D4^00gN?L_t(Y$IX^`P}F4@ z$D!$t#%ao_Y^J#WVWgVO6swt{(UQvn3?V{FaJ*nBzyNhogn?Z~4v|aFmGJrKV~6&3tBO_ucpVeV_01{C>L{8vmC|+m!z_ zv5C+pH2*Eptwbu(OZ-ku6SKq|(M*I98~*8OLz$Sln-Kh@N=W3LxHdS>QB>0fesU$u z-IAH<6k-2�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 RR}cUI002ovPDHLkV1miz7Bv6> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/xfce4-backdrop.png b/Media/Themes/Umami/Icon/apps/xfce4-backdrop.png new file mode 100644 index 0000000000000000000000000000000000000000..fe80dcee3060418d8fc581705cefc75733b16e85 GIT binary patch literal 1170 zcmV;D1a13?P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WdsL_t(Y$JLZgXk1kk z$A5PwnR&@`CQW9{Xw8^3LSsyAZ8a_gwSpyxU_eSiK_Q@47p^KH;!05vckYY}H`ax= z(hioo>Bf&VZ8bKLG&a+;O)`~srkzaZ^L@YTVqQXIrkPOOc;N8fJNKURyZ>`|7yj=i zR$o5-?31^E@&Bmq!LiW~z65Am7aSiR9!61G4bwE8+8l|LfRsoi(glg@#Xw4l?buRL z0%{_W7zb(q037-DXfTt>0B|N*BCiQ<<_)Bjb&ncrYhvH7F!6YtXlFZ-NaSDZHGoVe z!^p@8uIo}ca)v;&pOe2PDQiYuK-Prz2lsF-b)KOYo(EuNX6DWS0A1ISQnED`;qmQJ z=2AKm!KGBLE@WaZ$3O_Crlx8X`06E?rip2qs6ihs0f@E*80_yv2tbG$znCf@CG6h4 zhn}9Ejavj@Sr$@Cgpf$r;r`}*zH^9Mxgv{K7ILSsNWo%UuRP*s&ss4BEK zw!;3g11K$9Sa%5k;JPkCNG2yvQEmhX27~x~)uKX(wPy+F+wl}lEj=|k($xW&oH(_y zg08MErl+Ut%I}VVS(w8qFVq64|EJdgUVLe&dM(Iml9Y1oJAM5s-yi#KZSUYf5X18E zXHr{>ZR;CkcA-=k@I@k#c$-G%?*~A2{-T_}PDumkpzOtG7fRT+{nhGrafdQ3KzMZQ zy~$^uepbD@csXhqhR-z3YC9x^nP>130X~J_W;Nb@|8O^u_k80a1I<7%oz6@Yi-lMA z?tRX;va&*1dCLWOCjF0004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMza_)tt#MF0Q*000000000000000h>C@0X=NWth95|T zA4r2xQchA-P+?BaU|e2WU}9aQq@$jmot~bZpP!zs zt*jtQhcsoRHD#kVXQeo1qBv=$I%}vqY^XeFp+0Y`KyIi&aH~Ueu0(8|M|ZJPfwxsx zQ&?J7Sb(%yTUT0KSX+d)TwYpTc9&g?z+PZnUx&G2V_#usXJL=SVv4+DdXi;lV`YAv zWs16Glg4I_zG;rUX_v`rb9iczz-y4cY>~Kal*4XsY;JFDZg6aFkF;)^%x{yxZp2~T8cX^Jfe0_O*qRoAMdw!+Rfq{R4f`5Xj z(S(G9hpg0ziiwPxrjU}3lC;>Am6Vv6n4qAbprN0kp`fLvq^72(r>CZ5uCK4J zva_(YwY0Uiv$nRjx3{;rxVX8wxw^W#yu7@j&CAWr&Ck!z(9qD) z(b3e@)YR3~*V*0J*x1=_>=;`a}>FMg~ z>g((4?d|RD?(gpI?(gsK@9^>P@bK~R@$&NW^Yiob^Yrxe^!4@i_V)Jo_xJhv`TP6( z{QUg={r>;||F3er!~g&Q8+1}mQvd-10|Y5if{>J-vCq-e+1ugn@bUBX`~3X;&Y*6Z z00001VoOIvr@*%>0{{R4Pf0{UR47w*(dRpYQ5eVZCwuSRF|s*GBqMw8z4tCFk0`PV zk(II&Ndy0;>zvbbiRYMS!c7*bRzlNER8=rzD zCJ+@&$WH?pCy9eqGoc6!kdmB=)ia?K*fEfvfweQC3YamFMRYTv4m@L-z0e`p6Ng5i zN7Q8-uNIr_@$N!76zPRR7dX{uAcTAN&rie2^6W(Oeop`jec1p2_fj52~#yh z2ZXS*KaZA3BqYqVIpb~7G)=++F^6v}9#P0-G90`eyYcV!I-QPtpi-%h*5U8MEZ)3Y zv9rCxT|2ux77Cn#TOdDdDJD8DHY!vQ%I65;mIx#jYFfckWFqZQ;F#rGnC3Hnt zbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vhD=;uRFfc(bR}cUI002ovPDHLk FV1nvdCe#1` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/xfce4-ui.png b/Media/Themes/Umami/Icon/apps/xfce4-ui.png new file mode 100644 index 0000000000000000000000000000000000000000..019da41feb07d80bba9a883f8625275f4297ac05 GIT binary patch literal 1132 zcmV-y1e5!TP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00VAGL_t(Y$IX;oXj^3% z$A9NN=j5YFX3ME7rCC#|WizcxEt{*#&>N9WF9zO>b_ibhiCzv@@v?SK^v1j47}YTl zKd-7&9HlrUda)Xu);ToF>eAYxNt2x9oV@4tqG_6>iQ0kS1LxwL=Xw9X=lwqq2ma&G zwtJxO;MCi|c#HJe)XedtwI7TdXlnfa(cM4Y67B4hWf!OHQE()ee_fM5et&fwr~~}# z8}uEVItz>eJbCm%-gxbGT&XG&io-|7`E>Rh0NVWe#CIPan*tEq=zQ$a!w(P%`dL`a zvTy%BeBqn%MRs!Q?R!~~VPgF;9)Ii*g_3-^r9~%BPF6du509+hkI)4xnIf^lQC>Rq z5V{a7{!wH$nZ`7AO0KFq)occ1cMJ`?{r&xh5CWwXfV%i)LXe&z67EDpBbCmO`XWI` zInU~@-T3_hG+j_E6q^HHd39=1DQ^NG9*>)eM50c*(vi3_J zeRUDt&LjK*zRJV?yN9rCyADvDfl}VYGfzE7Z?q2~1U)@HNGWj~hpw(JmY0`_Mq@nt z^z%f6*D=$J2myxAz`v4a_|5_Bg1rS$XOUqTSeC`o(h{*)jFFKMq?Dx7X|7!Pf!xDcbAS_;p$X00svKNhXuz^Lggx=1@w}8;g<6W*8p63tiXQPSmZb zloHeQkrb>43$ngZ4bZqjL4pg^kTzN?Li zI+FHq007evtpRljAqcibP|BlVmz&l|&qLER1!%JulQkJ{p2|phpwxp zHL-yxfcEWtiqy>UM}XgeYrvoXcmp8>hGC$25?#nG`Jb;ZJOP{n$Q9F*z}1Ry4k!a^ z3&1c8y1TnEjICVDvhewQ)iwS6(ns%diT5|gQ4w_j7pU%5RY@r|clyk!1Dh}t%j(2+ z9oBM7oSA-e(_zp11X!yGYa89x3RkUec-Mg)z|IpVCl?Z5&)cC;$Q+6f7%v=qaWC*Y zunH8Ki8Y;NGy4G(@B_Y@&dNZk(SjQRH0Ek8Q2gKGFB0BSTjtApe*gdgC3HntbYx+4 zWjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hi yZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgGb=DKIxsLnE>{o$0000EzqmIn0Y3i@`+2_B{rzTd+x3zMz0$?i?M8yN3?`UZb zU|Mx>4S-oDPs9VB)mv}LC;%84f_M>7$^_Jvzy<*IEKo!Mb7=tC063m$?2ZcsV4i-I zABF85nqRbN08q~eq=kUB!;U7n*+>UW?g*oJ62aht=7rcfj&@{Zo~#lbt^S~o3ulyP zjaI2Hd0g2SLwg@7GRarVy*yjKQ;ip^n)!F@(t~t11B<6pJ5&o4U)R*{okdHHH$qa1 zhbjccbem0gwI=iH)WnxhA-P2b#QbKVKYg_#$vMD%O-JUb@hplD-;CkY%Kas$~O7IA2VW-_W1FV`HFa4mY<+R~ivKJN8qOZK;-GY0JV8QEpr&gxD&vHPTae1)4y;sXT z8)K*qzQvF(3VYW^2Y-7`KTl4ylZSKZtNi`zVWST`;~&52*>E(d&DFcM)n}ziZ%4X} zwdF3=&iM>_?B@ z;CDXSaBQQ-p$Gi8RyagxGE1PDI@c9a}{|?L>4GyN`;)E%2sAE zBv}duUnq)Ql6&U?@ZJgKGNppd_*;QPa0nI=!Xmg5b};E5NV2)U1VIwS_2pppe?xA* oJo|JBI0T!`BDgGekc7=4IYA`b|KGs#OQZpS@US@QNkm%3ANV2_W&i*H literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/ximian-evolution-calendar.png b/Media/Themes/Umami/Icon/apps/ximian-evolution-calendar.png new file mode 100644 index 0000000000000000000000000000000000000000..fc012d70718ad3d2ec600feed780d388afa1ae27 GIT binary patch literal 915 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wDhK$4xH1q8`~rMK!-GS^gM&f>PnTwGjAN=kWoc|}DRa#K4Zp= zdGqGYpFe-`;>AmrELpmA>C)wkmn~bieEISfD^{#rxpMXD)w_1>+O>Pf!Gi}6A3l8i z`0-PxPMtY(=E8*wmo8npeEIU#t5>gGyLRKoja#>F-M)SM&Ye4V@7}$4@813U_a8iX z@bKZoM~@yoe*E~!lP6D~K7ID=+4JYmU%q_#>eZ{)uV24;^XBc_x9{G)d;k9ZhYue< ze*F09)2Gj$KY#i1@)aE`bfTh!YgV5nevDFsr;D~52;9+RkEjabzSr&$v{uL{P z8D2OXy~5nUq_s?*;Y1|6*57v}qMj}}=iV)FaB*^B3YpO1qQssoqgq;%L-qNS&M3ZeXH3czGPZrGuJos*PHHE?f>=Z&iT{z z^Z&0%sGd5ZJaqm2JN306JzKZ9gf5ctzayu&e&wOfrrk#smqyq3AOHBRQNx;{;b+>? zq}}}*f-`sTKQ6fH8!y8JTa%VcQ(YV`-`cf?W0JsyjJ1pm3=RKc8tNLD=^C1c7#dj_nOhkeLo_(*g$sb9z|+;wWt~$( F69ALPcg+9* literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/apps/xscreensaver.png b/Media/Themes/Umami/Icon/apps/xscreensaver.png new file mode 100644 index 0000000000000000000000000000000000000000..0543d365d00e941b596f815a7383ddb0391411a3 GIT binary patch literal 1116 zcmV-i1f%004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Ul0L_t(Y$K8}&XdGn} zhM(EZ&Q21|n%%8bV%ntSqDhUW1Z+WTs`N&Rgv8v0qIjV;5aLZW2LA|_LJ>s05R8;U zZS_W4vo~4`mh{3Fp_{NqjhoQjh#{LaO|sp;`N_iav?0Ft8mOwl?~@PE_woe6z+C{M(J0f?(;Iqi z2{1Mm=cnsCx%JyQB0s1^qmyWwiKbcTxFvoqj3zQfmm!3Aq4Ng-NHMAzc4eiKsucPBpwPl^#@y7o8Z1XZ~$zCu(^3N?&>Y* zdE+Vz3uypkSpg)D9?{(!(%J+i?Ok*Im}~!RC3|c!_1_V?E(0Fek94!)%i+DlEloM!s*Ye zH{_h16jf+5K=b3xjQ={$cCD#tUugg*PjwN)mSQ0Qw~)154?HfnL}-r$?02RUDrK^VE`Zm1cTKyH9n>f zM~3G<`S{ZpfV=CwxJ0h`0RQ~_?Vq#RO!wi#&w7@YmdIo?7=})uDnK~2gIqrEPo+|` zfCA`%>XH`s1F!&7)3od7zrL{C*toaz(VE)eo;~$SusS47O-&h>f4rjRvbmh9YFB|3 z*R}P{?GCzD=Yj(vA&_zI8(7TgIa6=3yT zb|p~U-FN^KFkHq3H~yd6KYP@Km9iq~YXATMC3HntbYx+4WjbSWWnpw>05UK#FfA}S zEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tX iR53C-GB-LgGb=DKIxsLnE>{o$0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00i(!L_t(Y$Gw(qOjB1F z$D8c?{IYDBS+ZquV_C8#Q)7%-GM!^?$~NUiH;`$fvV?5P9Kp%z1m<8Ob52M*pzD{LBzd->o^e#t%e-t+&R^W1Y!E{F4^ zj!hqY8$n<-VpA*BpPek9CIW=*3~csM=zGmj_uWL!I0nB)a-+2m}I98?9h`0gOy8!aOk#m1z`h>Rz;KjhMbOgX!sMXa~k{ zx}j+?su$hQDF2oFy{Lko5v4Lj~UgCGF+2(9V7FwxC5|z0Xv1; zCYR7-^ME~A!3xZ{6Ut5q1SJ3$6V=sBE+7W){#l zu>b=zhYo|~p>USj*;!Q7wn8RXpsr(>T}=TJ;1NMwYQ|#qZmN5 z$&035FN6jUsuhD+T=YLAn3|eGm)?MsOg^gQW^f~u=0XXoK1#^-U6J)8yVi_hJ9EFY z_cn@Kj3{d#N0EX7Uu?&@@-}1?$Y zgAFHCAwRaLqXliJX%t1RW^@@$@Or($FZD-$4H9{s!QcZq3hd34ASvf6G#U+B#bTVl zti*}D7Mu-_Z(J6nhukjKzk4il-#0~fekxO;wyhV|p&$cE1?5IcvlA1NM#K2k*t~ zi$#H;1Tp*Kup#`wABSGw@?c4@YI$N}V%8Di#8Kh`A&iQOvPMTo-=Xs*0xS%}1X$?B z2H5s0;X$g_u08sfK&#cRA;O4Hh-9LWP!n!qv81E~X=!P!0MrKCPDn@yUZe8>U7I7U zL@9BQSWmq0;0)FfVZ8;fC;$Ke literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/applications-accessories.png b/Media/Themes/Umami/Icon/categories/applications-accessories.png new file mode 100644 index 0000000000000000000000000000000000000000..6725141da6261718453ef85be8568e8050e83ffb GIT binary patch literal 1471 zcmV;w1wi_VP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000XU000XU0RWnu7ytkO24YJ`L;(K){{a7>y{D4^00hBFL_t(Y$Gui-Op|96 zhF&N|8UR{ZbbK9r(*8Prkh0Ip=xK`n?>=7F_tDpBQi2mm$ws$OSo!&_^2BRNpY;1 zJDbQo2$Nfi5|iTw`_+9g7!0^^BLnFf*Du*3r@Z{4+xZwB9Yt4n7qVrUnhoSU)lGql zp0H)lqS0vN1Tr|?mq0{NOMy{tNGmFBl-G9)Uw`X~EUJj*9Xj8zK zVA1P!Tz3qNU_(4KAt{y;^zNhyQv|26KJ+q26W%XS#-hI}*I+q1yjsVJR0k&Jg*{0?O3%da$ zBioRXl@^!#M-DU^4TsXQ60l)gegC=Vub`#1r*fDlPs2Qa8#ZUKLEy8f=txXWO>$&? zeSIh>%0pvwBZh{C$R6!~gQf1@YrQ2|GR@>-butVqV;{l1u!DeAq@h?|w9af|q$Oo{ zP*-2e30_O(1{+G`QdJJ`$=pZ;OvVUI_l96Wdkkzvo3r^l3i#43)^z$|3=R%LtyZJF zq7>Els@eMX5{bR)Ccnwh{_(2mNeUJ;$I)Kkg!!TOu`-jr^9`&bk?FC&a|2}p^$m3> zNKW7il``H>-4zEHSEe(uFnkhyRbD36`HcEqEF06GPoTuX;Zo9fVW_0P2XtWg)-V76%eE2rEMT-BkDA&%;uiVj5_NQ39Bo_)TgGYkVax?^m zd_MPNXSq;-%014we<}$2%GW_wEfe{NoIv+mm^WOZ8`uz_j=v$|@g8q)XncJ!vVT7Z zBDXLB_I&n^#aUjJr`U4$q@dZlQ5gUG2<8mGZWWOG>^)H|6zVkl_Cev|GEHJL4?H~3 zE|r3A6)8XUNosMBT?0F__vLP4C|`C3%P$rR^sJr}SKC3J7+g-y6r~)xPbiu|AB+ebP zfq|%#NX$fjtjy7I(%|op3bA;U3f+`4d*fK=8C3haf_v}oh=z;8#k$?jV0C4OoYva% zM9aG7gP7_M-zrcd7C&a*c1We>78e(c1_Yqe$!WaULGUAONNs;@h-rH8d$gYOLbZ>C zi?h5A4=Fd33h!OeHok(PdVl&y$0M6IS#(b#xvX?`g_;J}<>|@D8PXIo`}6IE?n_ex z7hoK?h$;0CxPST$F3$2;KC9~rdk6n?e~q!OuV8rah3ze(4LVONT2m$x)zFZ?Wxf;4 zifntXU?Xb2q*V%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vi ZD=;uRFffFPX`TQ8002ovPDHLkV1f@GkUanZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/applications-development.png b/Media/Themes/Umami/Icon/categories/applications-development.png new file mode 100644 index 0000000000000000000000000000000000000000..2a9e81c49464468f85ad6e1f551f6fc3c894df15 GIT binary patch literal 1489 zcmV;?1upuDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00h%XL_t(Y$Gw$Xh+X9s z$A9bF``qTt%$dwga>j8o+DbHDFc`!p)!IvZ5bKQ;K}EqQ@j=83f^TZuhf+#K3Z;cU zj56AmKBQ37R%$6_&cs_2OVt{pp)_75t4w_?kOE{ObQqugzDF_gnuA?bWw5 z?Y^rvThgq*tZ9Emn@fMz#rfgo+8pL=8fQE_G@pNB>+ILQXrfqh<_O|mMf-n1(}QH~ zgCuLeXX?Uxx0$$Dugy{YuTxZ;BaRCr$BV-k&ThZ_VaBGfBP1^)B8Y&hMMICq$>WqK z_FB1jV4ChuTF-<&538 zyhdf}`db&DjlOwSz~VFJT2;Af_bm&c9bEb@!8Jh21XX;ncxMUTq9I^Rgld;47cqI> z4Ps2;m$f-be`HkGaM=k|9l@=lssx`Ad`j>M-t~#|d4f+68OC-GQy$yG zxtBdw7Gr*RT0rgD(eI4S+*lo%+==ZUCHNlRb#U27INQLx4#9QtzKhLT*GU0 z(m70LQGHl z*GDxa_!i#1gR{%1&q&&@Fn<1RX7z*L?)wY?7&VdL?JB_+F{VK99l+sT69hy-)q`4` zYofsoz#dgkd*ubjXZC=|u73g4<`{w?8WQ@Qw-J=!)=&cDGxy`OI^MTX*CO~+c;Cjm zl;!0nTPJg$0#rSPksY9cD|QmUaTFH=TyqcE;(RW|`Kc)B979AvB182)@^CzQV-z2MECp+Odbt8aUg=+14g+<7@{F z>n$663K)heSEB*a%q7hhLz}y3BwyK6Pg*A#8oiP&=iG&u!g^^#?@Vh3k&WrEFuL_- z;`|ie2I>tySgI2?7dblo`k@1r@!3bL>ya;18LI5X+tu}=p23Qx2}URPkSm^#;+U9u zh}Oq{yK^bY#a>-v#u!`ENCFkSBVQ%1cRcW zj`Ei4kZ6V@zus-X^|1X)y~#14xe-AJXr4I!#=>yr`D>>yd5C_iM%sB3k$5l@lZYuG zCXYydeLh4G4UBBv4;1;wYmaj1ul5J^Chr5?%@G)&3{1WF)7-(C3$EX>eb=Ly$dI&O zCG8y@K-S$v1#+c}D3&ipqG7tLZ*%hKk2&_%3k$bCsCNJjAOUo9?-^hSn0)Sge0Qdr z-+$f}j~9n3JEb`C1){iwiE@D8eM)dCr9ME!zW`1R{@M=2^>l#E zpa{qT#eo-q65xSsAe0QAv%&ifVde9(PXOPYiRy;pij4pO03~!qSaf7zbY(hYa%Ew3 zWdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWI rFflPLFg7hQH&ih)Ix;spF*GYMFgh?Wgo$aM00000NkvXXu0mjfn8LNM literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/applications-games.png b/Media/Themes/Umami/Icon/categories/applications-games.png new file mode 100644 index 0000000000000000000000000000000000000000..0266dbce481d486a9e305411753caba6d75b5c2d GIT binary patch literal 1261 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YWNL_t(Y$Gw%!OI%kN z#((GBkC{*!bp)L(S_4U=O(HHVNo~?VLo=nJWbdk!cA?n?iU@AZLL)*8!Bsa6?H^Fw zl(vu--8O2pA2EwKtwN$QIx0Fk_uli-h1YS$8L$M}7hcZ2mwTV*d7t+^=fZy+;@`eF zGCGt1I)Jx?5U*KlC;IyaetIc_kl4@uyo;~}9*VNSHPfY$2 zY;SLO_4f_T9!oGXI^+P|uIpZxLSBiKI^CE`70&LU8zh6054ALK05%LTbpOT`S$CtfL0ekYklRz4{o|)7^1aicXt=16xLccHa0kw zK1CEoD5WR|tF;Ots5@DQF(wV*0!S(ITI;4LiqKj!Ha3P*ik_aHSgF7mgE6K;X*ux0 zbzE7={sV%kLZNWe_kE_Prx_d^L~G6D>Xb~{;^y$2J{k``|r_=QI_EKM8kLP)%opL}DEiJKq?;dt<582d&T3o~` z#VfaO<6pa0ZquPf{e1&lVHB|l#UR5ptZ0&Eaa zeE6_x2+A!&2r;$0TflLgBl8Z{9+vK0xPU1ZF+s4u77D1ZBZOk*}t+uGMS{Ip@HV+W_-`DX^{}3 z{9r=}ft0fJ+}i3U1I%O>o^3y#ok`8-mJ z<&sb3F)})I#u)Qau~_8!^A}I@>v=9;zH}eKxpK@J*YN=)EP@xSE32P>`NjS3j{E7$ zxJp0HM4U;$0Td7dLD?F`srf(RAN1x{br!WjPyhe`C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML zHZ3tXR53C-GB-LgG%GMLIxsMViD{ky000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* XE^l&Yo9;Xs00000NkvXXu0mjf`vD|P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/applications-graphics.png b/Media/Themes/Umami/Icon/categories/applications-graphics.png new file mode 100644 index 0000000000000000000000000000000000000000..7310646200b3a8e415947c46b7b05cbd179b0344 GIT binary patch literal 1055 zcmV+)1mOFLP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 iR53C-GB-LgG%GMLIxsMViD{ky0000F? literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/applications-office.png b/Media/Themes/Umami/Icon/categories/applications-office.png new file mode 100644 index 0000000000000000000000000000000000000000..26be59e36f3521bf938a4abe476d63187a6ee643 GIT binary patch literal 1338 zcmV-A1;zS_P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/gnome-devel.png b/Media/Themes/Umami/Icon/categories/gnome-devel.png new file mode 100644 index 0000000000000000000000000000000000000000..2a9e81c49464468f85ad6e1f551f6fc3c894df15 GIT binary patch literal 1489 zcmV;?1upuDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00h%XL_t(Y$Gw$Xh+X9s z$A9bF``qTt%$dwga>j8o+DbHDFc`!p)!IvZ5bKQ;K}EqQ@j=83f^TZuhf+#K3Z;cU zj56AmKBQ37R%$6_&cs_2OVt{pp)_75t4w_?kOE{ObQqugzDF_gnuA?bWw5 z?Y^rvThgq*tZ9Emn@fMz#rfgo+8pL=8fQE_G@pNB>+ILQXrfqh<_O|mMf-n1(}QH~ zgCuLeXX?Uxx0$$Dugy{YuTxZ;BaRCr$BV-k&ThZ_VaBGfBP1^)B8Y&hMMICq$>WqK z_FB1jV4ChuTF-<&538 zyhdf}`db&DjlOwSz~VFJT2;Af_bm&c9bEb@!8Jh21XX;ncxMUTq9I^Rgld;47cqI> z4Ps2;m$f-be`HkGaM=k|9l@=lssx`Ad`j>M-t~#|d4f+68OC-GQy$yG zxtBdw7Gr*RT0rgD(eI4S+*lo%+==ZUCHNlRb#U27INQLx4#9QtzKhLT*GU0 z(m70LQGHl z*GDxa_!i#1gR{%1&q&&@Fn<1RX7z*L?)wY?7&VdL?JB_+F{VK99l+sT69hy-)q`4` zYofsoz#dgkd*ubjXZC=|u73g4<`{w?8WQ@Qw-J=!)=&cDGxy`OI^MTX*CO~+c;Cjm zl;!0nTPJg$0#rSPksY9cD|QmUaTFH=TyqcE;(RW|`Kc)B979AvB182)@^CzQV-z2MECp+Odbt8aUg=+14g+<7@{F z>n$663K)heSEB*a%q7hhLz}y3BwyK6Pg*A#8oiP&=iG&u!g^^#?@Vh3k&WrEFuL_- z;`|ie2I>tySgI2?7dblo`k@1r@!3bL>ya;18LI5X+tu}=p23Qx2}URPkSm^#;+U9u zh}Oq{yK^bY#a>-v#u!`ENCFkSBVQ%1cRcW zj`Ei4kZ6V@zus-X^|1X)y~#14xe-AJXr4I!#=>yr`D>>yd5C_iM%sB3k$5l@lZYuG zCXYydeLh4G4UBBv4;1;wYmaj1ul5J^Chr5?%@G)&3{1WF)7-(C3$EX>eb=Ly$dI&O zCG8y@K-S$v1#+c}D3&ipqG7tLZ*%hKk2&_%3k$bCsCNJjAOUo9?-^hSn0)Sge0Qdr z-+$f}j~9n3JEb`C1){iwiE@D8eM)dCr9ME!zW`1R{@M=2^>l#E zpa{qT#eo-q65xSsAe0QAv%&ifVde9(PXOPYiRy;pij4pO03~!qSaf7zbY(hYa%Ew3 zWdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWI rFflPLFg7hQH&ih)Ix;spF*GYMFgh?Wgo$aM00000NkvXXu0mjfn8LNM literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/gnome-globe.png b/Media/Themes/Umami/Icon/categories/gnome-globe.png new file mode 100644 index 0000000000000000000000000000000000000000..29c229031fc60d79d15d41b4fcfbbf636bd55870 GIT binary patch literal 1532 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 XE^l&Yo9;Xs00000NkvXXu0mjf`vD|P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/gnome-multimedia.png b/Media/Themes/Umami/Icon/categories/gnome-multimedia.png new file mode 100644 index 0000000000000000000000000000000000000000..41374e91b5385e8351e78ac3f9680bbdf132d544 GIT binary patch literal 1428 zcmV;F1#9|=P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00fpvL_t(Y$Gw$ZOcZAn zhM(D)otb4OsTSnr*~~=z zhgDTo@pwFpkB@Wp>QyvNqqepdUDqjDw>}Y?no#|IzW`9?0q#*0`K`=lnd`E%GYecL zB^kN7xpuGDYxVhj9654?{QdjUb)BA`9t^{vwY8O@KmX$X{Xy~OrYchibH+5yt~B@p z1-08f3?9vYA6I{Ske2uunA!^uSA zQX-l7DH#d>>i7F&&z{7b0?8sVJv}YC-EIuS;Le>pG&eWX?ekH({*6R#enI?`qo4Oj zqmiFy!r@26qTl-ce)DB@nhJHDFI>35)~#FRD_5=%3bke z$g(9r4JxODyp_%m_P6ZQR8^JEpFdAH9LDSQa^S!Lo;-O%Fc`#cx8riTsH&>MFbwYA zy-Q6^4coSD~_Yh*5S(7daOi9mp^j_;9# zNnmUYdxj0W%|^@Ky%-}8$*}3X-O|Fyg9l{jIt42WlB44R%a}3pTnzwlIP6SCW^gzh zw6?Y)gupaS3JVJ((ubi;Ho&Tr@Q`0WdHyfXCy(<#I7FFo33ML?RIsSzdr?d7i*ZEEc1$uaD~LY7QMb z#PQ?DxqbUKeSLkLIB|mE;bG$OI8LXN!ootXUAsm+9!Cg)u4#!S71(s0WHQN)9XmLE z`ZR908-S^)DeCI#04ORdnp5F+yV<;XGu_?Y0BD-Vk_5V@A%x)K#f#64eY#V8J|6(Z z#l^I=v@EEBX_`wDq?05`S`>VKJUBQw7pDp4=PgQLu~?|AtXvfM+4cOdO~>JISS&@* z_1)PY-{IBF3Us9x(hCwCKsK;EnYdf zJx{9^KrH3a6j6LW`2S-60Bol7hyPSuZ2$lOC3HntbYx+4WjbSWWnpw>05UK#FfA}S zEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tX iR53C-GB-LgG%GMLIxsMViD{ky0000F? literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/gnome-other.png b/Media/Themes/Umami/Icon/categories/gnome-other.png new file mode 100644 index 0000000000000000000000000000000000000000..e976f2f671287f746530b368f8977ecfc8b6b0d6 GIT binary patch literal 1130 zcmV-w1eN=VP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00V4EL_t(Y$IX*pXj@en zz<=jE_a-+nTieXm{+Z6zwrR6D+Uh_QhfV}R=!*#32)?>uhfWZtLs3CxY{PB?1sxS! zeG!J>iw`;xVN9)Nx;5=obYo)MbzPHHmnJv4IX?8(Rh$0Fi+JGUyAS95e&6?i?A`arN{_wo{1629|bCY^e}+PiiUygcscS1@x0(HguQ>2D4NTbCJ^EC!G5JeW=o zAFnQ5A4gjO#17*$bYn&8(Aq+4jVk3yfA?mvwfAstF^QC~{6IQAa=gaOOdwigN9qx+ zDNg-}P(lYIh_(pbwgoK~x$!}#^{KwH614#ScDfcl1kL z_~}EG9U@SF7rM3+M1%RKK%yOHc;7;J+27P}tugh|L)wmZ6qOZ3jEBkmjM5gO1SQHJ zi$X5}U-Y5PNrqqFW;wAZT+udG?IuJil&D#WQbns?7|&@Sqx+Vq?6YaX?eSa)2tJcGo7lk_k-y+rw_*)N{rg?l;BXz9=OV zm_2uMKd=_K1*ieSD-i{Jcs+1C5bJ4XYchdxFML`oUi!==6PWEC^Z=be9B2j_fhZ7~ z4^w3XwqHru-y8-)Th?-G-vgWgckNk2Sl1KMhzL>QD001R)MObuXVRU6W zV{&C-bY%cCFfuSLFgYzSHdHY| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s83o-U3d9>?EK-I%9jDA2awkZ-{PYXPB> zJ&s+~4K^$t|5(Hq?KX<-veqnJ@@0Ak?~-K;o>W#alz4UOZSX7HZP#3WGi|AfWM;}~ z<9X*l-=6gF@V(nC33l`C-<9poc93WdTCC-{^N_~iI3LjroM<@^yul+{^1;p$cgN|GMq*4ZEm3>r82aX~ z-D-F??ex*Lz0G>XjK9{*((LD-;g>$sXxW5` zOJ2OrbSZ4D0|NttYKdz^NlIc#s#S7PYGO$$gOP!efv$n2u7O#Ip@o&Psg;fq{X+)78&qol`;+0L4%lE&u=k literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/gnome-system.png b/Media/Themes/Umami/Icon/categories/gnome-system.png new file mode 100644 index 0000000000000000000000000000000000000000..7640af1f20b33940d3dca638daa68a0bba8f4f32 GIT binary patch literal 1532 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 ZD=;uRFffFPX`TQ8002ovPDHLkV1f@GkUanZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/gtk-preferences.png b/Media/Themes/Umami/Icon/categories/gtk-preferences.png new file mode 100644 index 0000000000000000000000000000000000000000..c9a892cc54ab9bfa4c826ba99f6dd2f1e2d91908 GIT binary patch literal 708 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s83o-U3d9>?EK-I%9jDA2awkZ-{PYXPB> zJ&s+~4K^$t|5(Hq?KX<-veqnJ@@0Ak?~-K;o>W#alz4UOZSX7HZP#3WGi|AfWM;}~ z<9X*l-=6gF@V(nC33l`C-<9poc93WdTCC-{^N_~iI3LjroM<@^yul+{^1;p$cgN|GMq*4ZEm3>r82aX~ z-D-F??ex*Lz0G>XjK9{*((LD-;g>$sXxW5` zOJ2OrbSZ4D0|NttYKdz^NlIc#s#S7PYGO$$gOP!efv$n2u7O#Ip@o&Psg;fq{X+)78&qol`;+0L4%lE&u=k literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/input_devices_settings.png b/Media/Themes/Umami/Icon/categories/input_devices_settings.png new file mode 100644 index 0000000000000000000000000000000000000000..db3f26200dbadcb2297617a1b5ab0f70c7747008 GIT binary patch literal 1397 zcmV-*1&aEKP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00cHkL_t(Y$GwzWY?MV5 z$A90QeAUhzX%wD%V#H1iU0BXh@WphzW|)hz4U~#H?a#f+>h;LNq*q z#!v{+2e0uG0`WpV+|@4RIicE9Tah)L@;*H3C|Yh``i+drh!J<|rVeT9EZ z-q6r+*I+JldgBvMjb1H`_yFuld+sR?_I=P8lTxMuR^R^spMUiwAMW30rb@1qB9qDR z{Bv7*@BQ7rXf!%f3NUWDY)-`DF}-%pI$^ZOLu(%dV=#a*25EZnm`-y`GtH-40C-_b zEqnGhu&}zC*0wgMltc%1)+OHn@H`~}OA3kWy5vXl-vK`^d)fByi3xh#X(L9*FF zE~hTB{?W%ca`Z4um();R5v8{;L#n$Q*L5krhLS~2CX;&|e_%z&**`#i&5GqUgn~ih zGvX-a0r1Ut-*WDJ2cb}iu8UoSLm_;QgAgz<&_6Db4e(}i$Lm6f9Y_ly1cecg>II8P zbzh?W+*vMMJdg4`Y+;i~Bv`d-1&==b2nDx5DwSe*cz8@2PouTQ?*x!i{sXXWd&j0{ zHXA9W6$*u@udhcbh2sxmVew+^%Ve`zGJU-qK70sc3_iaPX-lfAswfm(yiu3F{(j2B z5qv%$wzMZoRB&CZx3`yhbLRq(N~Ok60g&qMCKihUkjeDlAgd)Lkx-cFv1tSY0i1v{ zbq%&}-_EXGyT-?pt)e1Yx;rbPNFfnIVB2rGe!Pbt74g^j`B z^ZAjsjj#j-*Tpv#pf8i5FO#9Htc+QS1g&jtHykt=icFZhqb^!&^0^%O;XLVd50_J^ z)e`_i*&MNH)5n8Gy-|Ap>Y3mh%H{w#`svYP-2gm{$s1!*j^nf&W6o=>TedvA^@|CB zNF+Qag#{KC#%Pq*Xrs|ep|ob^9Ww#g^z>$no3vslpVY>nj6!RLQXWbxwAN^)(b{0N zF2Id>C@M8-ZVGxlW}W9aI-z?Ca0fc>`v7#OEu|>eL5_`LVmB8Sw0RD6P8*$#~-2U6brT_o{HFQN-bVF}#ZDnqB04QTA zATls8G$2Z0Yjt8EQ*>o%Ze?-`3PW;bVRU6=Aa`kWXdqN*WgtgMO;C{8i*En`03~!q zSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuX zVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*GYMFgh?Wgo$aM00000NkvXXu0mjf D$dhc7 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/kcontrol.png b/Media/Themes/Umami/Icon/categories/kcontrol.png new file mode 100644 index 0000000000000000000000000000000000000000..c9a892cc54ab9bfa4c826ba99f6dd2f1e2d91908 GIT binary patch literal 708 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s83o-U3d9>?EK-I%9jDA2awkZ-{PYXPB> zJ&s+~4K^$t|5(Hq?KX<-veqnJ@@0Ak?~-K;o>W#alz4UOZSX7HZP#3WGi|AfWM;}~ z<9X*l-=6gF@V(nC33l`C-<9poc93WdTCC-{^N_~iI3LjroM<@^yul+{^1;p$cgN|GMq*4ZEm3>r82aX~ z-D-F??ex*Lz0G>XjK9{*((LD-;g>$sXxW5` zOJ2OrbSZ4D0|NttYKdz^NlIc#s#S7PYGO$$gOP!efv$n2u7O#Ip@o&Psg;fq{X+)78&qol`;+0L4%lE&u=k literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/package_development.png b/Media/Themes/Umami/Icon/categories/package_development.png new file mode 100644 index 0000000000000000000000000000000000000000..2a9e81c49464468f85ad6e1f551f6fc3c894df15 GIT binary patch literal 1489 zcmV;?1upuDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00h%XL_t(Y$Gw$Xh+X9s z$A9bF``qTt%$dwga>j8o+DbHDFc`!p)!IvZ5bKQ;K}EqQ@j=83f^TZuhf+#K3Z;cU zj56AmKBQ37R%$6_&cs_2OVt{pp)_75t4w_?kOE{ObQqugzDF_gnuA?bWw5 z?Y^rvThgq*tZ9Emn@fMz#rfgo+8pL=8fQE_G@pNB>+ILQXrfqh<_O|mMf-n1(}QH~ zgCuLeXX?Uxx0$$Dugy{YuTxZ;BaRCr$BV-k&ThZ_VaBGfBP1^)B8Y&hMMICq$>WqK z_FB1jV4ChuTF-<&538 zyhdf}`db&DjlOwSz~VFJT2;Af_bm&c9bEb@!8Jh21XX;ncxMUTq9I^Rgld;47cqI> z4Ps2;m$f-be`HkGaM=k|9l@=lssx`Ad`j>M-t~#|d4f+68OC-GQy$yG zxtBdw7Gr*RT0rgD(eI4S+*lo%+==ZUCHNlRb#U27INQLx4#9QtzKhLT*GU0 z(m70LQGHl z*GDxa_!i#1gR{%1&q&&@Fn<1RX7z*L?)wY?7&VdL?JB_+F{VK99l+sT69hy-)q`4` zYofsoz#dgkd*ubjXZC=|u73g4<`{w?8WQ@Qw-J=!)=&cDGxy`OI^MTX*CO~+c;Cjm zl;!0nTPJg$0#rSPksY9cD|QmUaTFH=TyqcE;(RW|`Kc)B979AvB182)@^CzQV-z2MECp+Odbt8aUg=+14g+<7@{F z>n$663K)heSEB*a%q7hhLz}y3BwyK6Pg*A#8oiP&=iG&u!g^^#?@Vh3k&WrEFuL_- z;`|ie2I>tySgI2?7dblo`k@1r@!3bL>ya;18LI5X+tu}=p23Qx2}URPkSm^#;+U9u zh}Oq{yK^bY#a>-v#u!`ENCFkSBVQ%1cRcW zj`Ei4kZ6V@zus-X^|1X)y~#14xe-AJXr4I!#=>yr`D>>yd5C_iM%sB3k$5l@lZYuG zCXYydeLh4G4UBBv4;1;wYmaj1ul5J^Chr5?%@G)&3{1WF)7-(C3$EX>eb=Ly$dI&O zCG8y@K-S$v1#+c}D3&ipqG7tLZ*%hKk2&_%3k$bCsCNJjAOUo9?-^hSn0)Sge0Qdr z-+$f}j~9n3JEb`C1){iwiE@D8eM)dCr9ME!zW`1R{@M=2^>l#E zpa{qT#eo-q65xSsAe0QAv%&ifVde9(PXOPYiRy;pij4pO03~!qSaf7zbY(hYa%Ew3 zWdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWI rFflPLFg7hQH&ih)Ix;spF*GYMFgh?Wgo$aM00000NkvXXu0mjfn8LNM literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/package_games.png b/Media/Themes/Umami/Icon/categories/package_games.png new file mode 100644 index 0000000000000000000000000000000000000000..0266dbce481d486a9e305411753caba6d75b5c2d GIT binary patch literal 1261 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YWNL_t(Y$Gw%!OI%kN z#((GBkC{*!bp)L(S_4U=O(HHVNo~?VLo=nJWbdk!cA?n?iU@AZLL)*8!Bsa6?H^Fw zl(vu--8O2pA2EwKtwN$QIx0Fk_uli-h1YS$8L$M}7hcZ2mwTV*d7t+^=fZy+;@`eF zGCGt1I)Jx?5U*KlC;IyaetIc_kl4@uyo;~}9*VNSHPfY$2 zY;SLO_4f_T9!oGXI^+P|uIpZxLSBiKI^CE`70&LU8zh6054ALK05%LTbpOT`S$CtfL0ekYklRz4{o|)7^1aicXt=16xLccHa0kw zK1CEoD5WR|tF;Ots5@DQF(wV*0!S(ITI;4LiqKj!Ha3P*ik_aHSgF7mgE6K;X*ux0 zbzE7={sV%kLZNWe_kE_Prx_d^L~G6D>Xb~{;^y$2J{k``|r_=QI_EKM8kLP)%opL}DEiJKq?;dt<582d&T3o~` z#VfaO<6pa0ZquPf{e1&lVHB|l#UR5ptZ0&Eaa zeE6_x2+A!&2r;$0TflLgBl8Z{9+vK0xPU1ZF+s4u77D1ZBZOk*}t+uGMS{Ip@HV+W_-`DX^{}3 z{9r=}ft0fJ+}i3U1I%O>o^3y#ok`8-mJ z<&sb3F)})I#u)Qau~_8!^A}I@>v=9;zH}eKxpK@J*YN=)EP@xSE32P>`NjS3j{E7$ zxJp0HM4U;$0Td7dLD?F`srf(RAN1x{br!WjPyhe`C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML zHZ3tXR53C-GB-LgG%GMLIxsMViD{ky000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* XE^l&Yo9;Xs00000NkvXXu0mjf`vD|P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/package_graphics.png b/Media/Themes/Umami/Icon/categories/package_graphics.png new file mode 100644 index 0000000000000000000000000000000000000000..7310646200b3a8e415947c46b7b05cbd179b0344 GIT binary patch literal 1055 zcmV+)1mOFLP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00R6;L_t(Y$GwzaXj4}l z$3OQ^dwZ{GL)+BVl^|MLTPv~KwDf{6Iv65?Fnp1|?8zYN7zm|q2;D@mY%gO_2yC#8 zl`+PoFFx%-e9)i|N=kG#iHiz0O_Q6X32Az7(tCYrp_H_Vw0hvcxxah9pU*w#at^%X zChQc*uPcfG---gpqBCCH0QQ0@au{O z+}YRN)A_ki6K7s7^6bSz5t!UK_pJqfU6Fy?o;`beP9Ji~;pi)#8Zyg;j1By=Z2|C` z+tWGpnXf}BCL=_aKOmP%EQ9?+AZQl1CGhLYgsY|PjNb3nKCC?F@w`OdR>`F1ZQJ3; zYMHtPeqFhwsIH5Bz1_QX_e&nci@aFr!pXlj1?UaZz(Mz3w~E-&v`Yomlig!3zCNo4c9F)0w4` zX`K!P%{!arYpy`k8c#JfDe8&ipHe6k1jmM$bCyi#FKeY@{jn9iu74oN($_;rhTRPf z4aDPdmX=EF?(A~J-jlYq!dJEOTM8y7uJ_|Ohkd@oBoYad$t032I);(Xic0fGfuLEc zCD?k4xV6KMCtKC{RE^h?CxAq2(3?>N%)}Dp{001R)MObuXVRU6WV{&C-bY%cC zFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00fpvL_t(Y$Gw$ZOcZAn zhM(D)otb4OsTSnr*~~=z zhgDTo@pwFpkB@Wp>QyvNqqepdUDqjDw>}Y?no#|IzW`9?0q#*0`K`=lnd`E%GYecL zB^kN7xpuGDYxVhj9654?{QdjUb)BA`9t^{vwY8O@KmX$X{Xy~OrYchibH+5yt~B@p z1-08f3?9vYA6I{Ske2uunA!^uSA zQX-l7DH#d>>i7F&&z{7b0?8sVJv}YC-EIuS;Le>pG&eWX?ekH({*6R#enI?`qo4Oj zqmiFy!r@26qTl-ce)DB@nhJHDFI>35)~#FRD_5=%3bke z$g(9r4JxODyp_%m_P6ZQR8^JEpFdAH9LDSQa^S!Lo;-O%Fc`#cx8riTsH&>MFbwYA zy-Q6^4coSD~_Yh*5S(7daOi9mp^j_;9# zNnmUYdxj0W%|^@Ky%-}8$*}3X-O|Fyg9l{jIt42WlB44R%a}3pTnzwlIP6SCW^gzh zw6?Y)gupaS3JVJ((ubi;Ho&Tr@Q`0WdHyfXCy(<#I7FFo33ML?RIsSzdr?d7i*ZEEc1$uaD~LY7QMb z#PQ?DxqbUKeSLkLIB|mE;bG$OI8LXN!ootXUAsm+9!Cg)u4#!S71(s0WHQN)9XmLE z`ZR908-S^)DeCI#04ORdnp5F+yV<;XGu_?Y0BD-Vk_5V@A%x)K#f#64eY#V8J|6(Z z#l^I=v@EEBX_`wDq?05`S`>VKJUBQw7pDp4=PgQLu~?|AtXvfM+4cOdO~>JISS&@* z_1)PY-{IBF3Us9x(hCwCKsK;EnYdf zJx{9^KrH3a6j6LW`2S-60Bol7hyPSuZ2$lOC3HntbYx+4WjbSWWnpw>05UK#FfA}S zEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tX iR53C-GB-LgG%GMLIxsMViD{ky0000F? literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/package_network.png b/Media/Themes/Umami/Icon/categories/package_network.png new file mode 100644 index 0000000000000000000000000000000000000000..29c229031fc60d79d15d41b4fcfbbf636bd55870 GIT binary patch literal 1532 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 ZD=;uRFffFPX`TQ8002ovPDHLkV1f@GkUanZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/preferences-desktop-peripherals.png b/Media/Themes/Umami/Icon/categories/preferences-desktop-peripherals.png new file mode 100644 index 0000000000000000000000000000000000000000..db3f26200dbadcb2297617a1b5ab0f70c7747008 GIT binary patch literal 1397 zcmV-*1&aEKP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00cHkL_t(Y$GwzWY?MV5 z$A90QeAUhzX%wD%V#H1iU0BXh@WphzW|)hz4U~#H?a#f+>h;LNq*q z#!v{+2e0uG0`WpV+|@4RIicE9Tah)L@;*H3C|Yh``i+drh!J<|rVeT9EZ z-q6r+*I+JldgBvMjb1H`_yFuld+sR?_I=P8lTxMuR^R^spMUiwAMW30rb@1qB9qDR z{Bv7*@BQ7rXf!%f3NUWDY)-`DF}-%pI$^ZOLu(%dV=#a*25EZnm`-y`GtH-40C-_b zEqnGhu&}zC*0wgMltc%1)+OHn@H`~}OA3kWy5vXl-vK`^d)fByi3xh#X(L9*FF zE~hTB{?W%ca`Z4um();R5v8{;L#n$Q*L5krhLS~2CX;&|e_%z&**`#i&5GqUgn~ih zGvX-a0r1Ut-*WDJ2cb}iu8UoSLm_;QgAgz<&_6Db4e(}i$Lm6f9Y_ly1cecg>II8P zbzh?W+*vMMJdg4`Y+;i~Bv`d-1&==b2nDx5DwSe*cz8@2PouTQ?*x!i{sXXWd&j0{ zHXA9W6$*u@udhcbh2sxmVew+^%Ve`zGJU-qK70sc3_iaPX-lfAswfm(yiu3F{(j2B z5qv%$wzMZoRB&CZx3`yhbLRq(N~Ok60g&qMCKihUkjeDlAgd)Lkx-cFv1tSY0i1v{ zbq%&}-_EXGyT-?pt)e1Yx;rbPNFfnIVB2rGe!Pbt74g^j`B z^ZAjsjj#j-*Tpv#pf8i5FO#9Htc+QS1g&jtHykt=icFZhqb^!&^0^%O;XLVd50_J^ z)e`_i*&MNH)5n8Gy-|Ap>Y3mh%H{w#`svYP-2gm{$s1!*j^nf&W6o=>TedvA^@|CB zNF+Qag#{KC#%Pq*Xrs|ep|ob^9Ww#g^z>$no3vslpVY>nj6!RLQXWbxwAN^)(b{0N zF2Id>C@M8-ZVGxlW}W9aI-z?Ca0fc>`v7#OEu|>eL5_`LVmB8Sw0RD6P8*$#~-2U6brT_o{HFQN-bVF}#ZDnqB04QTA zATls8G$2Z0Yjt8EQ*>o%Ze?-`3PW;bVRU6=Aa`kWXdqN*WgtgMO;C{8i*En`03~!q zSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuX zVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*GYMFgh?Wgo$aM00000NkvXXu0mjf D$dhc7 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/preferences-desktop.png b/Media/Themes/Umami/Icon/categories/preferences-desktop.png new file mode 100644 index 0000000000000000000000000000000000000000..c9a892cc54ab9bfa4c826ba99f6dd2f1e2d91908 GIT binary patch literal 708 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s83o-U3d9>?EK-I%9jDA2awkZ-{PYXPB> zJ&s+~4K^$t|5(Hq?KX<-veqnJ@@0Ak?~-K;o>W#alz4UOZSX7HZP#3WGi|AfWM;}~ z<9X*l-=6gF@V(nC33l`C-<9poc93WdTCC-{^N_~iI3LjroM<@^yul+{^1;p$cgN|GMq*4ZEm3>r82aX~ z-D-F??ex*Lz0G>XjK9{*((LD-;g>$sXxW5` zOJ2OrbSZ4D0|NttYKdz^NlIc#s#S7PYGO$$gOP!efv$n2u7O#Ip@o&Psg;fq{X+)78&qol`;+0L4%lE&u=k literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/preferences-system.png b/Media/Themes/Umami/Icon/categories/preferences-system.png new file mode 100644 index 0000000000000000000000000000000000000000..a04819c3106cd61aefbbcc7aa566fddb5babb335 GIT binary patch literal 1143 zcmb7@eK6Yx7{{M1wDq>GYFhfjMo};GOPU40+Bz2TZbUj&E%6d*M~Mn$lly68oNY#G z9XeuM$XH2Py_~mJYnwS`ox5SI>ZL-wG(_da7CUqMV}JH@_r3c*chBd~Z+>u)w~@ZN zJ^&c``cP;9aCj&C@BlDc_hU=l$pWda=AjGu&k^sFE39fleM(8Xti2PzdRCCFwtZ%L=Hsx};d_%du_FS>PrckVJY;0_9zECMQRVt-Q zrBbVuYPC|WR%tY9jYhMzwWZZ+x3{;4+5D3L;K`u{g#zvsypHakz4``*`;8Bpn3)|t zcFe-s+Q!z--ofcD1VIo2fk-5h(bH$nP$+)>!5@aDW4s(uS6BZpgCoyXnysS#wsru( z#rS(uil^~YtpK=7z7!9-T_{(;3Vrb2Yq>d|Cc4HB=?9yXPnLui`(Af2UG9!?&cE11 zEx^WnA0YQ#SH+#u%xg)k!fI{9M}74-?&(DO9d{g2Cv_nA7;DcS=X@1kDnO&M!@jV4 zqsGQCpbLYRi4b&HCGK z@)g~a9b5KXAD;Y6M{m@p3A5=jJ1}t~BtvMGeEx#8BvIFo^-aZ-;Kq$UZzg}Gw?`-&Ns_5?@0N@12>l`O8n!`k6uQH=KOduje#0hb8LdbN2D|+f}lt6Ys z5EMbaR=5)WCtOL6yBwbhM1(+aLdZ@87dnB65?xRNX;004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000XU000XU0RWnu7ytkO24YJ`L;(K){{a7>y{D4^00hBFL_t(Y$Gui-Op|96 zhF&N|8UR{ZbbK9r(*8Prkh0Ip=xK`n?>=7F_tDpBQi2mm$ws$OSo!&_^2BRNpY;1 zJDbQo2$Nfi5|iTw`_+9g7!0^^BLnFf*Du*3r@Z{4+xZwB9Yt4n7qVrUnhoSU)lGql zp0H)lqS0vN1Tr|?mq0{NOMy{tNGmFBl-G9)Uw`X~EUJj*9Xj8zK zVA1P!Tz3qNU_(4KAt{y;^zNhyQv|26KJ+q26W%XS#-hI}*I+q1yjsVJR0k&Jg*{0?O3%da$ zBioRXl@^!#M-DU^4TsXQ60l)gegC=Vub`#1r*fDlPs2Qa8#ZUKLEy8f=txXWO>$&? zeSIh>%0pvwBZh{C$R6!~gQf1@YrQ2|GR@>-butVqV;{l1u!DeAq@h?|w9af|q$Oo{ zP*-2e30_O(1{+G`QdJJ`$=pZ;OvVUI_l96Wdkkzvo3r^l3i#43)^z$|3=R%LtyZJF zq7>Els@eMX5{bR)Ccnwh{_(2mNeUJ;$I)Kkg!!TOu`-jr^9`&bk?FC&a|2}p^$m3> zNKW7il``H>-4zEHSEe(uFnkhyRbD36`HcEqEF06GPoTuX;Zo9fVW_0P2XtWg)-V76%eE2rEMT-BkDA&%;uiVj5_NQ39Bo_)TgGYkVax?^m zd_MPNXSq;-%014we<}$2%GW_wEfe{NoIv+mm^WOZ8`uz_j=v$|@g8q)XncJ!vVT7Z zBDXLB_I&n^#aUjJr`U4$q@dZlQ5gUG2<8mGZWWOG>^)H|6zVkl_Cev|GEHJL4?H~3 zE|r3A6)8XUNosMBT?0F__vLP4C|`C3%P$rR^sJr}SKC3J7+g-y6r~)xPbiu|AB+ebP zfq|%#NX$fjtjy7I(%|op3bA;U3f+`4d*fK=8C3haf_v}oh=z;8#k$?jV0C4OoYva% zM9aG7gP7_M-zrcd7C&a*c1We>78e(c1_Yqe$!WaULGUAONNs;@h-rH8d$gYOLbZ>C zi?h5A4=Fd33h!OeHok(PdVl&y$0M6IS#(b#xvX?`g_;J}<>|@D8PXIo`}6IE?n_ex z7hoK?h$;0CxPST$F3$2;KC9~rdk6n?e~q!OuV8rah3ze(4LVONT2m$x)zFZ?Wxf;4 zifntXU?Xb2q*V%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vi ZD=;uRFffFPX`TQ8002ovPDHLkV1f@GkUanZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/redhat-games.png b/Media/Themes/Umami/Icon/categories/redhat-games.png new file mode 100644 index 0000000000000000000000000000000000000000..0266dbce481d486a9e305411753caba6d75b5c2d GIT binary patch literal 1261 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YWNL_t(Y$Gw%!OI%kN z#((GBkC{*!bp)L(S_4U=O(HHVNo~?VLo=nJWbdk!cA?n?iU@AZLL)*8!Bsa6?H^Fw zl(vu--8O2pA2EwKtwN$QIx0Fk_uli-h1YS$8L$M}7hcZ2mwTV*d7t+^=fZy+;@`eF zGCGt1I)Jx?5U*KlC;IyaetIc_kl4@uyo;~}9*VNSHPfY$2 zY;SLO_4f_T9!oGXI^+P|uIpZxLSBiKI^CE`70&LU8zh6054ALK05%LTbpOT`S$CtfL0ekYklRz4{o|)7^1aicXt=16xLccHa0kw zK1CEoD5WR|tF;Ots5@DQF(wV*0!S(ITI;4LiqKj!Ha3P*ik_aHSgF7mgE6K;X*ux0 zbzE7={sV%kLZNWe_kE_Prx_d^L~G6D>Xb~{;^y$2J{k``|r_=QI_EKM8kLP)%opL}DEiJKq?;dt<582d&T3o~` z#VfaO<6pa0ZquPf{e1&lVHB|l#UR5ptZ0&Eaa zeE6_x2+A!&2r;$0TflLgBl8Z{9+vK0xPU1ZF+s4u77D1ZBZOk*}t+uGMS{Ip@HV+W_-`DX^{}3 z{9r=}ft0fJ+}i3U1I%O>o^3y#ok`8-mJ z<&sb3F)})I#u)Qau~_8!^A}I@>v=9;zH}eKxpK@J*YN=)EP@xSE32P>`NjS3j{E7$ zxJp0HM4U;$0Td7dLD?F`srf(RAN1x{br!WjPyhe`C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML zHZ3tXR53C-GB-LgG%GMLIxsMViD{ky000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* XE^l&Yo9;Xs00000NkvXXu0mjf`vD|P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/redhat-graphics.png b/Media/Themes/Umami/Icon/categories/redhat-graphics.png new file mode 100644 index 0000000000000000000000000000000000000000..7310646200b3a8e415947c46b7b05cbd179b0344 GIT binary patch literal 1055 zcmV+)1mOFLP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/redhat-programming.png b/Media/Themes/Umami/Icon/categories/redhat-programming.png new file mode 100644 index 0000000000000000000000000000000000000000..2a9e81c49464468f85ad6e1f551f6fc3c894df15 GIT binary patch literal 1489 zcmV;?1upuDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00h%XL_t(Y$Gw$Xh+X9s z$A9bF``qTt%$dwga>j8o+DbHDFc`!p)!IvZ5bKQ;K}EqQ@j=83f^TZuhf+#K3Z;cU zj56AmKBQ37R%$6_&cs_2OVt{pp)_75t4w_?kOE{ObQqugzDF_gnuA?bWw5 z?Y^rvThgq*tZ9Emn@fMz#rfgo+8pL=8fQE_G@pNB>+ILQXrfqh<_O|mMf-n1(}QH~ zgCuLeXX?Uxx0$$Dugy{YuTxZ;BaRCr$BV-k&ThZ_VaBGfBP1^)B8Y&hMMICq$>WqK z_FB1jV4ChuTF-<&538 zyhdf}`db&DjlOwSz~VFJT2;Af_bm&c9bEb@!8Jh21XX;ncxMUTq9I^Rgld;47cqI> z4Ps2;m$f-be`HkGaM=k|9l@=lssx`Ad`j>M-t~#|d4f+68OC-GQy$yG zxtBdw7Gr*RT0rgD(eI4S+*lo%+==ZUCHNlRb#U27INQLx4#9QtzKhLT*GU0 z(m70LQGHl z*GDxa_!i#1gR{%1&q&&@Fn<1RX7z*L?)wY?7&VdL?JB_+F{VK99l+sT69hy-)q`4` zYofsoz#dgkd*ubjXZC=|u73g4<`{w?8WQ@Qw-J=!)=&cDGxy`OI^MTX*CO~+c;Cjm zl;!0nTPJg$0#rSPksY9cD|QmUaTFH=TyqcE;(RW|`Kc)B979AvB182)@^CzQV-z2MECp+Odbt8aUg=+14g+<7@{F z>n$663K)heSEB*a%q7hhLz}y3BwyK6Pg*A#8oiP&=iG&u!g^^#?@Vh3k&WrEFuL_- z;`|ie2I>tySgI2?7dblo`k@1r@!3bL>ya;18LI5X+tu}=p23Qx2}URPkSm^#;+U9u zh}Oq{yK^bY#a>-v#u!`ENCFkSBVQ%1cRcW zj`Ei4kZ6V@zus-X^|1X)y~#14xe-AJXr4I!#=>yr`D>>yd5C_iM%sB3k$5l@lZYuG zCXYydeLh4G4UBBv4;1;wYmaj1ul5J^Chr5?%@G)&3{1WF)7-(C3$EX>eb=Ly$dI&O zCG8y@K-S$v1#+c}D3&ipqG7tLZ*%hKk2&_%3k$bCsCNJjAOUo9?-^hSn0)Sge0Qdr z-+$f}j~9n3JEb`C1){iwiE@D8eM)dCr9ME!zW`1R{@M=2^>l#E zpa{qT#eo-q65xSsAe0QAv%&ifVde9(PXOPYiRy;pij4pO03~!qSaf7zbY(hYa%Ew3 zWdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWI rFflPLFg7hQH&ih)Ix;spF*GYMFgh?Wgo$aM00000NkvXXu0mjfn8LNM literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/redhat-sound_video.png b/Media/Themes/Umami/Icon/categories/redhat-sound_video.png new file mode 100644 index 0000000000000000000000000000000000000000..41374e91b5385e8351e78ac3f9680bbdf132d544 GIT binary patch literal 1428 zcmV;F1#9|=P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00fpvL_t(Y$Gw$ZOcZAn zhM(D)otb4OsTSnr*~~=z zhgDTo@pwFpkB@Wp>QyvNqqepdUDqjDw>}Y?no#|IzW`9?0q#*0`K`=lnd`E%GYecL zB^kN7xpuGDYxVhj9654?{QdjUb)BA`9t^{vwY8O@KmX$X{Xy~OrYchibH+5yt~B@p z1-08f3?9vYA6I{Ske2uunA!^uSA zQX-l7DH#d>>i7F&&z{7b0?8sVJv}YC-EIuS;Le>pG&eWX?ekH({*6R#enI?`qo4Oj zqmiFy!r@26qTl-ce)DB@nhJHDFI>35)~#FRD_5=%3bke z$g(9r4JxODyp_%m_P6ZQR8^JEpFdAH9LDSQa^S!Lo;-O%Fc`#cx8riTsH&>MFbwYA zy-Q6^4coSD~_Yh*5S(7daOi9mp^j_;9# zNnmUYdxj0W%|^@Ky%-}8$*}3X-O|Fyg9l{jIt42WlB44R%a}3pTnzwlIP6SCW^gzh zw6?Y)gupaS3JVJ((ubi;Ho&Tr@Q`0WdHyfXCy(<#I7FFo33ML?RIsSzdr?d7i*ZEEc1$uaD~LY7QMb z#PQ?DxqbUKeSLkLIB|mE;bG$OI8LXN!ootXUAsm+9!Cg)u4#!S71(s0WHQN)9XmLE z`ZR908-S^)DeCI#04ORdnp5F+yV<;XGu_?Y0BD-Vk_5V@A%x)K#f#64eY#V8J|6(Z z#l^I=v@EEBX_`wDq?05`S`>VKJUBQw7pDp4=PgQLu~?|AtXvfM+4cOdO~>JISS&@* z_1)PY-{IBF3Us9x(hCwCKsK;EnYdf zJx{9^KrH3a6j6LW`2S-60Bol7hyPSuZ2$lOC3HntbYx+4WjbSWWnpw>05UK#FfA}S zEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tX iR53C-GB-LgG%GMLIxsMViD{ky0000F? literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/redhat-system_settings.png b/Media/Themes/Umami/Icon/categories/redhat-system_settings.png new file mode 100644 index 0000000000000000000000000000000000000000..a04819c3106cd61aefbbcc7aa566fddb5babb335 GIT binary patch literal 1143 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/xfce-games.png b/Media/Themes/Umami/Icon/categories/xfce-games.png new file mode 100644 index 0000000000000000000000000000000000000000..0266dbce481d486a9e305411753caba6d75b5c2d GIT binary patch literal 1261 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00YWNL_t(Y$Gw%!OI%kN z#((GBkC{*!bp)L(S_4U=O(HHVNo~?VLo=nJWbdk!cA?n?iU@AZLL)*8!Bsa6?H^Fw zl(vu--8O2pA2EwKtwN$QIx0Fk_uli-h1YS$8L$M}7hcZ2mwTV*d7t+^=fZy+;@`eF zGCGt1I)Jx?5U*KlC;IyaetIc_kl4@uyo;~}9*VNSHPfY$2 zY;SLO_4f_T9!oGXI^+P|uIpZxLSBiKI^CE`70&LU8zh6054ALK05%LTbpOT`S$CtfL0ekYklRz4{o|)7^1aicXt=16xLccHa0kw zK1CEoD5WR|tF;Ots5@DQF(wV*0!S(ITI;4LiqKj!Ha3P*ik_aHSgF7mgE6K;X*ux0 zbzE7={sV%kLZNWe_kE_Prx_d^L~G6D>Xb~{;^y$2J{k``|r_=QI_EKM8kLP)%opL}DEiJKq?;dt<582d&T3o~` z#VfaO<6pa0ZquPf{e1&lVHB|l#UR5ptZ0&Eaa zeE6_x2+A!&2r;$0TflLgBl8Z{9+vK0xPU1ZF+s4u77D1ZBZOk*}t+uGMS{Ip@HV+W_-`DX^{}3 z{9r=}ft0fJ+}i3U1I%O>o^3y#ok`8-mJ z<&sb3F)})I#u)Qau~_8!^A}I@>v=9;zH}eKxpK@J*YN=)EP@xSE32P>`NjS3j{E7$ zxJp0HM4U;$0Td7dLD?F`srf(RAN1x{br!WjPyhe`C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML zHZ3tXR53C-GB-LgG%GMLIxsMViD{ky000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* XE^l&Yo9;Xs00000NkvXXu0mjf`vD|P literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/xfce-graphics.png b/Media/Themes/Umami/Icon/categories/xfce-graphics.png new file mode 100644 index 0000000000000000000000000000000000000000..7310646200b3a8e415947c46b7b05cbd179b0344 GIT binary patch literal 1055 zcmV+)1mOFLP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 iR53C-GB-LgG%GMLIxsMViD{ky0000F? literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/xfce-office.png b/Media/Themes/Umami/Icon/categories/xfce-office.png new file mode 100644 index 0000000000000000000000000000000000000000..26be59e36f3521bf938a4abe476d63187a6ee643 GIT binary patch literal 1338 zcmV-A1;zS_P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00cZqL_t(Y$F-ANY*bYg zhW~xeT+Up2rt~rcy-iCkMOvW}r1+#r1WUme(*$E9B-&a@tkBR%V@xAK^(KidaeQwswv(LY(q-fQpue_3l~1OKxK z05p8k^`WGpSu-ddi3&M4I_j;*F?(|t|DK{*XF9t@eIdsHfT*JRz)Ks9H}*W0Syqsl zdZzJQaHZz0b$O>`i$A>B{2zct5%;*A{a1QNwYu%=oW-j%hN_#+l`Pl1egE3y^$m^9 zFFwC1vh<7r0G>n7+4H}y?HigJ>K&Lw&9-8;$e(d%OGBsUQRYp%`7L{1sc*?Hu5fCK zH1j|}N88?40pza9bmq-U`rPF0$ z004v#geQbNAubA2f=Y)b1njB~dW!tsNILVT8h%q{MMaeq`31dwk$V$U$rDTO*cNZf z3w%dIV~KTyVUqSWnGRVs$o8iTbL}5oitH$i-2@>vYFW^AUC|9gCu83Q170;_C>+C_ zsC0LBi}ADoo0w!N2m4t1tQ51JN=6xA3J1R4D!vk0#C5s%knymj(2!r{Ar z4i1KfZ(hAH8T8v+vFJDiGxNUT-UnEK_*~>Q-J~SXC1fO18U@3mcOZ(&wK=TY<55mu z4CTjT!{0{wE_#)j0o)$Z}93zh{~1lhNWj^u0(BxP}WoVuMXZVgS5^}#IG%X{@N zuIeqN1-xf^P6T=4bq67gGabP8pkzne-peOWeT0_AeXhw_Q91w3S1y~Kz!-Na-y-N^w7IUI9ZmJbY3<*Tr`m7z5CAF2REjvn#5wUuRR{m%PXp2iT)5}?H=c_ z)n4ftCW0hGl9H%ouj|BHe5tH*n*)Hc>=`T(0RXSd5uubOSL=P$#xh`74nN1stlWb2 z!wkbX0BSxi003A3Vmvk~fm+B{d%$}fCrlu;Erl~ZQN?DR5S*cXYl4aSv z_S3b=W5?U$0A2vVFRg;40dN5F02Cf+J^V56<2N`Q4&Ty?s;Ua{cp~)Kr)N(B=mRhc zKz$ToA#a~A*$couzh?0F-(dh~^C2z@Z8GYFhfjMo};GOPU40+Bz2TZbUj&E%6d*M~Mn$lly68oNY#G z9XeuM$XH2Py_~mJYnwS`ox5SI>ZL-wG(_da7CUqMV}JH@_r3c*chBd~Z+>u)w~@ZN zJ^&c``cP;9aCj&C@BlDc_hU=l$pWda=AjGu&k^sFE39fleM(8Xti2PzdRCCFwtZ%L=Hsx};d_%du_FS>PrckVJY;0_9zECMQRVt-Q zrBbVuYPC|WR%tY9jYhMzwWZZ+x3{;4+5D3L;K`u{g#zvsypHakz4``*`;8Bpn3)|t zcFe-s+Q!z--ofcD1VIo2fk-5h(bH$nP$+)>!5@aDW4s(uS6BZpgCoyXnysS#wsru( z#rS(uil^~YtpK=7z7!9-T_{(;3Vrb2Yq>d|Cc4HB=?9yXPnLui`(Af2UG9!?&cE11 zEx^WnA0YQ#SH+#u%xg)k!fI{9M}74-?&(DO9d{g2Cv_nA7;DcS=X@1kDnO&M!@jV4 zqsGQCpbLYRi4b&HCGK z@)g~a9b5KXAD;Y6M{m@p3A5=jJ1}t~BtvMGeEx#8BvIFo^-aZ-;Kq$UZzg}Gw?`-&Ns_5?@0N@12>l`O8n!`k6uQH=KOduje#0hb8LdbN2D|+f}lt6Ys z5EMbaR=5)WCtOL6yBwbhM1(+aLdZ@87dnB65?xRNX;004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00jO?L_t(Y$IX^|Op{j_ z#vSU^ZJFCY+g!3`QM0(jMJHt#-PAaR%taD4x4N>$!Z-bKdto=cEf3 z{9m3;2TSg=GNhQny?-r6K8|Dv>q7~@9rW z@h2!;;_7CPU93kHF+X42#IzsHZ;&E*R7{!uN|O_`!;8+|5!m{MA!%~pyE+RJPf|$A z79j1c2J(dO)eZS zP$2D$2pMO^IDAfmBj=^~GN&GyxiVzs$?0(xFrqJK(8i(F; z9d*hcoFE2QC^Pitp#TcQ0JuULCrSUQLM5sMG(P=G=qKX2&n$9ud_~;h0+qjK;5J%1 zJy5l{plx+SDDUte%NIO3e@P0S*o<144GR4La*MSntT5nmxe}~Yp?AL2*g^$MC}=@T z_bA%BhM_XLP_OAnRh`ineTZi#YwGQ8_@7RWj{uN#Fh+5)VDQ_n)d%>8|1W&AaHdkqyx`3c11ofrv&8 zWtw$P0tVKeF(~y;V&Vp0(v9+~?Iq9 zv_UF!K;B?O4d3MFRBC;*4bmm@nfVh)?BG;)fXk!NVI4(LMI*@WGKWky?af`4e^L0y z_?;fJt^Z9Jg(YU?Y4gMNI7i#Oe1FIW!Pl7`kusf~He=v)K zGPW8M6H}O+{25Bc0Mtq+s;Ui$POkR+le?%RBTwbiXkCy>9f8%!W&N<(ZsGQ=`?xbc zg&RY6;dXu>pt)%Xtp*PYisd9q$*+-VLKcyEZcg#YbP4^*QTen=>+q?SE~pxv&?sDJ zQVv3=cB5G{1Op?ezkyNDkA$CKHrzyqaTx7JH_AB8Olqb#q)<^$nJ~`evU?(K?D3h5 z9_Tf0(B=_XEnXt%3D8c&EJTcM89}dg91~+xXlr(%j~<7kV;o|U86T!n{&n${8;NAZ zOnt^JBJvR56q_aWb+&t`@1b8u$#wVZy$q$9s{?Q_HRtCc^sK^ z^_NSWHlJE|1KBw>Pu3q~v)Aq~{fOu-4mVgtcxgidN53Ph0eeoVu`Z!%ko3i_*_|h5 z%-lVb=%?=lNX*q^<@U@{G9I(`K&ftP2G6&4Z;_C+R}x+(JRfebknkd5#hSSM)9aI} z2gvWOgq@5DW55*H*)pyb#RDg1Mqt%B&z_(URAfE?@CZFyz(me_o-Uc^F9{Wu$*q zaQtn;YlP*5WkK05UK#FfA}S zEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tX iR53C-GB-LgG%GMLIxsMViD{ky0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000XU000XU0RWnu7ytkO24YJ`L;(K){{a7>y{D4^00hBFL_t(Y$Gui-Op|96 zhF&N|8UR{ZbbK9r(*8Prkh0Ip=xK`n?>=7F_tDpBQi2mm$ws$OSo!&_^2BRNpY;1 zJDbQo2$Nfi5|iTw`_+9g7!0^^BLnFf*Du*3r@Z{4+xZwB9Yt4n7qVrUnhoSU)lGql zp0H)lqS0vN1Tr|?mq0{NOMy{tNGmFBl-G9)Uw`X~EUJj*9Xj8zK zVA1P!Tz3qNU_(4KAt{y;^zNhyQv|26KJ+q26W%XS#-hI}*I+q1yjsVJR0k&Jg*{0?O3%da$ zBioRXl@^!#M-DU^4TsXQ60l)gegC=Vub`#1r*fDlPs2Qa8#ZUKLEy8f=txXWO>$&? zeSIh>%0pvwBZh{C$R6!~gQf1@YrQ2|GR@>-butVqV;{l1u!DeAq@h?|w9af|q$Oo{ zP*-2e30_O(1{+G`QdJJ`$=pZ;OvVUI_l96Wdkkzvo3r^l3i#43)^z$|3=R%LtyZJF zq7>Els@eMX5{bR)Ccnwh{_(2mNeUJ;$I)Kkg!!TOu`-jr^9`&bk?FC&a|2}p^$m3> zNKW7il``H>-4zEHSEe(uFnkhyRbD36`HcEqEF06GPoTuX;Zo9fVW_0P2XtWg)-V76%eE2rEMT-BkDA&%;uiVj5_NQ39Bo_)TgGYkVax?^m zd_MPNXSq;-%014we<}$2%GW_wEfe{NoIv+mm^WOZ8`uz_j=v$|@g8q)XncJ!vVT7Z zBDXLB_I&n^#aUjJr`U4$q@dZlQ5gUG2<8mGZWWOG>^)H|6zVkl_Cev|GEHJL4?H~3 zE|r3A6)8XUNosMBT?0F__vLP4C|`C3%P$rR^sJr}SKC3J7+g-y6r~)xPbiu|AB+ebP zfq|%#NX$fjtjy7I(%|op3bA;U3f+`4d*fK=8C3haf_v}oh=z;8#k$?jV0C4OoYva% zM9aG7gP7_M-zrcd7C&a*c1We>78e(c1_Yqe$!WaULGUAONNs;@h-rH8d$gYOLbZ>C zi?h5A4=Fd33h!OeHok(PdVl&y$0M6IS#(b#xvX?`g_;J}<>|@D8PXIo`}6IE?n_ex z7hoK?h$;0CxPST$F3$2;KC9~rdk6n?e~q!OuV8rah3ze(4LVONT2m$x)zFZ?Wxf;4 zifntXU?Xb2q*V%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vi ZD=;uRFffFPX`TQ8002ovPDHLkV1f@GkUanZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/categories/xfce4-settings.png b/Media/Themes/Umami/Icon/categories/xfce4-settings.png new file mode 100644 index 0000000000000000000000000000000000000000..c9a892cc54ab9bfa4c826ba99f6dd2f1e2d91908 GIT binary patch literal 708 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4mJh`h6m-gKNuJoSkfJR9T^xl_H+M9WMyDr zP)PO&@?~JCQe$9fXklRZ#lXPO@PdJ%)PRBERRRNp)eHs(@q#(K0&N%=7}%1$-CY>| zgW!U_%O^81FmM)lL>4nJFnEBm+sSM@1_s83o-U3d9>?EK-I%9jDA2awkZ-{PYXPB> zJ&s+~4K^$t|5(Hq?KX<-veqnJ@@0Ak?~-K;o>W#alz4UOZSX7HZP#3WGi|AfWM;}~ z<9X*l-=6gF@V(nC33l`C-<9poc93WdTCC-{^N_~iI3LjroM<@^yul+{^1;p$cgN|GMq*4ZEm3>r82aX~ z-D-F??ex*Lz0G>XjK9{*((LD-;g>$sXxW5` zOJ2OrbSZ4D0|NttYKdz^NlIc#s#S7PYGO$$gOP!efv$n2u7O#Ip@o&Psg;fq{X+)78&qol`;+0L4%lE&u=k literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/clipboard.png b/Media/Themes/Umami/Icon/clipboard.png new file mode 100644 index 0000000000000000000000000000000000000000..317b3cd31564cfe38d2d39a9654fbdec81acf078 GIT binary patch literal 1196 zcmV;d1XKHoP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x1H(x~K~zY`omAaVTtyr`ckkU#cG=2)K+qz>8W+1YNJ&({ z6^sT&G`_aR=Qin6eK9dw;~!x9Wa7J#hxWllt1$+(Ek&BcR~8q(TweZt-nQ+#`zDxv`IYykUYb5pC={&WBSV>VI&B&T zF!kbON3mF#SzBA zm$tSfMu#ehMJ+gvt6wXuCIZDuf@^~IOa^^D2ho+!`PC5l6aGqiN*I8isbm7lwiJka z!I51QKrl@J!I4d;(B78xL1l2EezAd&^mhcVJ_XbJ-4$>FxHXXEA;`raE^qlk|5`z) z9y#o*5*$zpfL0kWKr3JYCZRg~m|uCl>iQmmRh;?nEa_X+Is*wX7bHYz$EBaHXh5T} zvVdF>R0qpJE>c@dEiKY0=iy*a~zFn>$jiIob2x=~a zCnPwN;Pv1DeE;(`NGY|7(Mn10w*?-wugcBgQe!BrW(6LD$0b}&zFh&(69at#5J7MO z8RoF1NR5b6z)52$tR@26V_IU3NeIq;&x6Z#y9^jx{uq@4g<(ix_u2p?>)xg2w#Qc?xH~X=a7kW$DhT9E3luTFm_Y`&b$pz#3CK{ zBke3B9gF;L@Jwe9Y(a3m<}R|jt=d;VUcWIt(9;Vm*T1joZNtFQuYaP;-2Bckj06!m zmT8*3rfEKK;*+cIzx|x(nq9du=nBKy4?)}{-kTJQZ_HkMCmM|o3nBjc2MI5xk5=yN zoB#j-C3HntbYx+4WjbSWWnpw>05UK#FfA}SEigA!FflqcH99deD=;uRFfcy*tZo1R z03~!qSaf7zbY(hiZ)9m^c>ppnGB7PLIV~_ZR4_3*H8eUiH7hVMIxsLN=0+a?0000< KMNUMnLSTYe5fOd> literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/default.png b/Media/Themes/Umami/Icon/default.png new file mode 100644 index 0000000000000000000000000000000000000000..ac9f7ec7d71b2861c694dfe0ac1ecfb88987ea82 GIT binary patch literal 535 zcmeAS@N?(olHy`uVBq!ia0y~yU=RRd4rT@h1`S>QUNSs54@ z6p}rHd>I(3)EF2VS{N99F)%PRykKA`HDF+PmB7GYHG_dcykO3*KpO@I2KEw9Usv`g zoNPh{2L0~)Z!j=0R0sHkxc>kDp8*VHT6gi5tdMEl#a^|AqiPFB^%l>zAGNgIQM4Oxi_;fyjyqU)BXqFuU@-$ z?b_9A*RS8Z|KR?;d(WP~c>e79+jsBYzkB=T%h&H;zyALH`~TlR|NsAwnD{QAfq{Xo zB*-tALB)8(!CNN}{5sFTz`&T~?e3zncf;0A3=9lOo-U3d65*+Rfm{s+JkFE1DIEIn z@AwyI7FKD&lDt{__e@ed;QnRNwRM5VQob6w6g_udV(9#ZEns5s_u{@IO1on>r7*;0 z&v9V*_ASoA_JaH$CBA*$jONbj$vl-N-+N;jE^uD-R7~31v-k}I1A}UbYeY#(Vo9o1 za#3nxNh*Vpfsui(fw``cafqR%m8pf5k-4^kft7)Q%en92C>nC}Q!>*kaceN0?Ejj9 xfk6^vLvVgtNqJ&XDuZK6ep0G}XKrG8YEWuoN@d~6R8VX&c)I$ztaD0e0st%l(4PPR literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/3floppy_unmount.png b/Media/Themes/Umami/Icon/devices/3floppy_unmount.png new file mode 100644 index 0000000000000000000000000000000000000000..dc1d789207c3f881b2511423e3260557db9e2bea GIT binary patch literal 980 zcmV;_11tQAP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00OZ|L_t(Y$Gw%kYZOrw z#eZ*iCdjH0W3-5%M6mIr@}o9b2}1q=6>LN+EhC76AXtbXVqsLQg0*0!C~9L9(8k6> z(2$@k2-#>>-pjmuEM{kR*IhL+=!M~Z&7AY@x$nGze;f|r;OPe!&Fpe92Gr+W{b5|r zIWU|%H+|>W?F7KgE}lO*=F+6ffydXt%prk8Y5S3t%@LP&hU;$jr>liX}1`G3#4o>&l>7t^S=vCSr`e62)kkX~lrO1V-$@=_s-x zGfrK9m75fDm!jOTtrpZ;FXky-EkVUOvMlxxxU40S4E*OeG(S9F){fU~J47Bg6DHC8_=55AaRvtmNz;@hNl4SQ>?cV=2!VRN&eYTtwV_=UX`9sF2QkYptzsv` z!^0#=!sO&+`SQfX1m1g^%_boPoXam?i!tx#8ugWk7$c^(*=n^KX_|J)(P%UXArL~q zdrt_VC>Amypkm6eou6w^fJ<$H&f^BKu&~etUJ&FNRFH%LS?rTRmKo4C+v4J4#}0o- z;PQ>R)<40~dz(7=I$Lk7AK6r6w*IXI^WOImbb^=bG9*_X?f7#Bg2Qj#e|olO`v5!G>*L z9#-2c-63_m-Qk$ocobZ%I65;mIx#jYFfckW zFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbzi zVk!Ut02y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00b#XL_t(Y$JLeJZxmG& z$3OSZot@bq-PKqM+bPSE(AYG_7qC2P(D)Pl2aJgXUyKrX_5maM2SCsmqX{DZ0lfHX zqV0>?2;EX2pauf%Lfd6`ySts8+4#x4~<*5ixoj&$;^>D%BK1RQdM*hdZWe*4pj%zHByo3=<(l;(;GLwSFYqrrBZHka*}j9Jvof0_0{`5%d%eFvhC4@ zg$hz?=z0pJ6iO*HO-t6pu}-H$tJT7?tYBf`nmlyq;PRh;%r64SBo(DjzVh;s1B;ak zsZpHrwlS-u?al3+*-`-0p zQA#0%h@}w8I=)Y{*~GG}5-|EW4MeZkOMtcslv3n!IWn0HQ550(KD}O#AP7)O4Gvn@8Xq5L_wL=S ztgNuUzK-KKgkgvff+&gzf&kz5iK2)?p@3L*Zu9*eypgcz&UYWTjN zxM3K$uFKZe)<9e-MMR{>IC|^?i2Bcjn#&z9+m|++Z z0@rcuk!G{`8{h*j5JKOu13W-qT)bY{zrQ^4pkABlb{(XYf$O@PZqL=aU0dt*Tx@&G zvu(TWI9+YaZfm~psYatY_v59@p8_>t6^I@uGQcii^o=)PKQ=Qn^M0$f`2ssw7(~@h zr*r$_#UH)|>OeIH4|nRKZPS5gc{sFkaRZpKP9g^I04MI2H6A{r&M?pu-{c|DQh*VI z`#>)y={*I;e`o&!nDW7rw3yDj0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>% zI65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(b dH##vjD=;uRFfbziVk!Ut002ovPDHLkV1l)0VnYA` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/audio-input-microphone.png b/Media/Themes/Umami/Icon/devices/audio-input-microphone.png new file mode 100644 index 0000000000000000000000000000000000000000..58f1a80c27f384d1cbb6ca6fce0de39e9feda719 GIT binary patch literal 1088 zcmZXTc}&v>9L2vK%mEG4DMZl02(gG{hx+BXdg>mZ!B!msdO#ulbnCOgIn8}i5FYo>J@_GNga!y33rNtQw z0I*~-Xi)%Qa}+ZZ0GQKGi~@kEoElCA_^KNFO8hneI7k^$;Q+Z#0EMLhF9D!%6Ce{0 zU=#r$^#G^SKYJWW0RZ!K#zz?dmUBnJpw5li0GLS*n@5NLi7(*t1zZ47sZ{@#tne^e z2r81q0f1hwHy8{6U~g~F7jPNOkYGBB(x`#d0KR}55J;f}kjWIkhK2^QRKyo>FMS+C z@h88T4Wb5yu^D^;H(J0CV>9^zE{Ah5UL4C}Guh!Rkt8mP8_5@NRXG{ze}o33!NI{= zT^oWRC`v_9G$e!`8X6MH2w^fAEEbE!W=2GWb0Q<6ct>D#3@_$Vv_vYACP-xRL|IaT zEGbcbWLc6zu1HQ&q{tL0^5o>?Pf}CT(o?QvT)vudB~z7=m7STLot2lDn}03$+VyJ% z1=ov;3X6)0ic1U1%F1q3lvP%hSKh3ws;a84u2!qn_4Rifo9dfd8k;put*xzJ+-ufo zG;M8dZJnC-_V$j>cCA*c)wSt#_jNj5cXwA$Pfu@e?}NUc{{H@ffq_B&;P8lHbZm5N zY;=5ld}4B9a%ysVdV2cF)XdDx{QUgF!ouR>;?mO6^78V^%F62M>e|}c`uh6D#>VF6 z=GNAh(P-S>-rm{S+1uOO-`_ttI5<2!tTrh>3jn^#s0c1#-ZC|}IDXQ~+V;%Z_djsJ z;cyo|B>9plfk9N179&j0$}TFY{j{;UqvOH%h9`@wYrA_dUcP$$`;Kis0AM7nP+Hv| zCX4+5mQ8x5V zJ{F5*y~8tK+j93mb^f{6Y(lAKtJBmtVzSjcG))lydG7~9gU!YBWJ{%8S#;2n&`Ml! zu9f(8SJrarv*!k{IkQktFT1)7w?bqg^~*jr&WZi`(^*eNR6$>*f5omx*Qw-W7uV4W zuCSPc`>ye?+{fP{#@~d?k+`Z@k%JxoCp~@G%&pi%{p=*C7G~eW){5jMFBSs*U3y)9 z9`tdW9jb}sRm7iYr(-+4TO2%H$q)6y_{cajr^(UXTAQ8I%PZDK>s?33Ve^M2w^_g7 zNF}^Sl89uxvjR^ke;ha10^DK@Uzq&0|12klp}}uO%)=-(=DwXx*r&_-`y5xpM#j literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/battery.png b/Media/Themes/Umami/Icon/devices/battery.png new file mode 100644 index 0000000000000000000000000000000000000000..9db15e1b19e3393c4745fde72d7992ee448043fe GIT binary patch literal 1214 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/camera.png b/Media/Themes/Umami/Icon/devices/camera.png new file mode 100644 index 0000000000000000000000000000000000000000..597cfa06fb33f9a96b923b356371e7b540ea60b8 GIT binary patch literal 1410 zcmV-|1%3L7P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/cdwriter_unmount.png b/Media/Themes/Umami/Icon/devices/cdwriter_unmount.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/chardevice.png b/Media/Themes/Umami/Icon/devices/chardevice.png new file mode 100644 index 0000000000000000000000000000000000000000..5d5ac83f7b903118b6234c0178f0bcd99c66c03a GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 FV1kwbK)L_` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/dvd_unmount.png b/Media/Themes/Umami/Icon/devices/dvd_unmount.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-battery.png b/Media/Themes/Umami/Icon/devices/gnome-dev-battery.png new file mode 100644 index 0000000000000000000000000000000000000000..9db15e1b19e3393c4745fde72d7992ee448043fe GIT binary patch literal 1214 zcmV;v1VQ_WP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Y2DL_t(Y$GwzIXdGo2 z$A2@kyGi#Wn{=C`iAn9D)K)#&s(A3=M-f42+De)T1wBQfpb??q!7uQd9P}a{Jw&S( zEP}<5C`F=>G-->iO=1sqv$1Kiv!AmwJ0I`sVP}>#31k)YfrnvU-uM4|p8xaCz<(S{ zGn>YT)%SsSHZ6ZJIw9Wu*MRY1^#$MzFc^-+u0D7C<%8XapV{Turd*l7U`|b5Tv4Rc z27ER;AfD`^6L0iNac@qF^rRG-AEmSOh5U46w=Dnhv%Ir# z(f;JMUO6_T3^dij*pPBeQIzjr|7Zd9U%@Ug_rrNo+6G;zcARR3E4MOic%7VnMT$x_DPUM0@0Il(zdJ z)>j*s^l*j5G0T{x4fJbYvU+2ZvSHG3==tUWMx{y;z_XG0-y!|2UwH)6bTBFw#cYPM zVb;%--Ii?uTZTL%PH@xMPu|-NTExF+5}XoLb4W`^nAyLb(VZk|d4L&lqPY(86_4|U z0S4_TnkXSfCyHz(f=vm?dJY?oByEOS%E$OK@A0Us1;;f&)T-JF&;W##2q9SY`iO+X zWV95PW8pe3j&R6Uvy`k7#cB~llzLkDr+BDZnKVV~UQ@2<@vpQBhT>?pnoko4V9 z8v&YcPo#0PX;zvN4{Y_LGni!~K&9cw7mdb5G#UfIvaB7&lgT7|I@&~omh`hu3i&og za=JdF7m6=FcI2q+@9$S!*G1Dp2qEfenkKew)85`e&)yzV9Z7jV@0?)QF4P9KjaZ~wi zaZ$NEJ#{XfS-T9RffBW=TDL{@{oe&7{9&gadie0MzOMMwZJqm`*v=Y#_4ZFI>&D#T zt>tOp4zLasg6bOq0YvjbQ9oONm=6x#HEMU<0t{cU0+juy6ZpRw&004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-cdrom.png b/Media/Themes/Umami/Icon/devices/gnome-dev-cdrom.png new file mode 100644 index 0000000000000000000000000000000000000000..cbf6f96e655c00cc6c637ced7e3cc5d784d7ed0d GIT binary patch literal 1176 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 JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-disc-cdrw.png b/Media/Themes/Umami/Icon/devices/gnome-dev-disc-cdrw.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdr-plus.png b/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdr-plus.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdr.png b/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdr.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdram.png b/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdram.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdrom.png b/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdrom.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdrw.png b/Media/Themes/Umami/Icon/devices/gnome-dev-disc-dvdrw.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-dvd.png b/Media/Themes/Umami/Icon/devices/gnome-dev-dvd.png new file mode 100644 index 0000000000000000000000000000000000000000..cbf6f96e655c00cc6c637ced7e3cc5d784d7ed0d GIT binary patch literal 1176 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 iR53C-GB-LgH7hVMIxsLI{$eTs0000 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-floppy.png b/Media/Themes/Umami/Icon/devices/gnome-dev-floppy.png new file mode 100644 index 0000000000000000000000000000000000000000..dc1d789207c3f881b2511423e3260557db9e2bea GIT binary patch literal 980 zcmV;_11tQAP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 DGb|cb literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-keyboard.png b/Media/Themes/Umami/Icon/devices/gnome-dev-keyboard.png new file mode 100644 index 0000000000000000000000000000000000000000..f9b6326b04d5d58ee3c879722fa249bef8eb3c78 GIT binary patch literal 890 zcmV-=1BLvFP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-removable-1394.png b/Media/Themes/Umami/Icon/devices/gnome-dev-removable-1394.png new file mode 100644 index 0000000000000000000000000000000000000000..96ac2d822a19a50ce36cb98b2a4862dd18509211 GIT binary patch literal 775 zcmV+i1Ni)jP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Id~L_t(Y$L*ERYtvvD z$3JhEuEliSplPx=d#Izkh@-puCwLG<(Bp1;D3l(=DfkcQX$%It2ov$3M?w5IcI>6B zr*>^t=`Rh+a;5YFI9kx>-ogIi`w5W( z&UN7E$>XQQhX5wkEcqz#~o57!HS#1xBM0`uYZ0McbVbxUFwIT#hf(@AnrHa2yBQzPKL} zC|62F`BsW#GKp!Lkp)cCB#}sAgdioS zXfztsYPD^@M8C~ncXV9`Ad^uR$^amr&(rPw%=!Yoeiy?q=yW<>cp$tfB4PrciSRBE z%I65;mIx#jYFfckWFqZQ;F#rGnC3Hnt zbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLk FV1kwbK)L_` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-removable-usb.png b/Media/Themes/Umami/Icon/devices/gnome-dev-removable-usb.png new file mode 100644 index 0000000000000000000000000000000000000000..96ac2d822a19a50ce36cb98b2a4862dd18509211 GIT binary patch literal 775 zcmV+i1Ni)jP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Id~L_t(Y$L*ERYtvvD z$3JhEuEliSplPx=d#Izkh@-puCwLG<(Bp1;D3l(=DfkcQX$%It2ov$3M?w5IcI>6B zr*>^t=`Rh+a;5YFI9kx>-ogIi`w5W( z&UN7E$>XQQhX5wkEcqz#~o57!HS#1xBM0`uYZ0McbVbxUFwIT#hf(@AnrHa2yBQzPKL} zC|62F`BsW#GKp!Lkp)cCB#}sAgdioS zXfztsYPD^@M8C~ncXV9`Ad^uR$^amr&(rPw%=!Yoeiy?q=yW<>cp$tfB4PrciSRBE z%I65;mIx#jYFfckWFqZQ;F#rGnC3Hnt zbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLk FV1kwbK)L_` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-removable.png b/Media/Themes/Umami/Icon/devices/gnome-dev-removable.png new file mode 100644 index 0000000000000000000000000000000000000000..96ac2d822a19a50ce36cb98b2a4862dd18509211 GIT binary patch literal 775 zcmV+i1Ni)jP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Id~L_t(Y$L*ERYtvvD z$3JhEuEliSplPx=d#Izkh@-puCwLG<(Bp1;D3l(=DfkcQX$%It2ov$3M?w5IcI>6B zr*>^t=`Rh+a;5YFI9kx>-ogIi`w5W( z&UN7E$>XQQhX5wkEcqz#~o57!HS#1xBM0`uYZ0McbVbxUFwIT#hf(@AnrHa2yBQzPKL} zC|62F`BsW#GKp!Lkp)cCB#}sAgdioS zXfztsYPD^@M8C~ncXV9`Ad^uR$^amr&(rPw%=!Yoeiy?q=yW<>cp$tfB4PrciSRBE z%I65;mIx#jYFfckWFqZQ;F#rGnC3Hnt zbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLk FV1kwbK)L_` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-dev-wavelan.png b/Media/Themes/Umami/Icon/devices/gnome-dev-wavelan.png new file mode 100644 index 0000000000000000000000000000000000000000..63c46e7b297ca2a9b82e11d70ef0f181edc0cd71 GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Uu3L_t(Y$Gw%$OI%kJ z$3N%Xd86Zu6O9rS#xOfEaToKG5VB~BLjHpag&222TP%S>3GG4GpY%}$w_O~z0&wsB_O?czO$slOH40~hW+_rCMF-}}DrKKPGAY6qiZ zBR#-pK>so8&8X^+cfR~;e#gw{*vPrWufNz_E^Ch)xm?yhc=(eYzxUloTXh39jL*;a zT)=fLtE;P?JQ1&DvzVnN^yN!*c^S8}6Zh;{D&5`K=4Pz509b43{kV^Z(+@rck^ou* z=x=Ll$8}vq1XV>#C7fiE;M5f6*)v2H_u@r#Z4Kw~V}j%31k!2L--xQx*49q#5eK#( zm8yD+i$zQ_2_HV--oA~E#SrmcSp~4Ji#~sjNu|)YZz=coVnw!>+X)a61kmMWXm7_} zx`Y_xx#8VAq)@<~I)y|caODc_nKS52B<2!wuTym?dd@S0eJr&O(x;R z3n&y2W3ZPmV{hKXHaEl7tLV}awxi>a3Q8r2$I(y-%4H;-MvQ@5w;&dSSFez{Ippb6 zY+oNX7K36DQ!E}5(9nQ{!(M-V9cgStZrlJ7D3zeQ8&03bCK6C6KuZg@wH3S$dhg1*O5rdi zzzKyw1Z`=7PzYB990%&^4pvb8xVF6QIA}PGitt-F%s1DslM98=%}wOwNhBKeHWLU` zZM-jFH;9Bnh$??%GW2U80R{4g;=T_l3~CeWh-_JXBS zDFz1z$z(E2PfxS3uuv_ySMhxnY|qWk&N4YU$;!$Kb8~YflgWJx@9CoLVXb9jV}sV# zR)&X%$!4=85(yR;7bz491cO29>+25%n91jJ{n6+N%H{IE<0Kgok{hd@42vbr6UJ9p4>SOcK-eEUfD8N$tOI`n8^B*c*`MPe0C)}) zALjT#%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q lEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1jFW+%^CJ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gnome-fs-client.png b/Media/Themes/Umami/Icon/devices/gnome-fs-client.png new file mode 100644 index 0000000000000000000000000000000000000000..2b73dada771072f676d2003ff47a1a52b327addb GIT binary patch literal 1215 zcmV;w1VH004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gtk-cdrom.png b/Media/Themes/Umami/Icon/devices/gtk-cdrom.png new file mode 100644 index 0000000000000000000000000000000000000000..8965f76c2ff13fe81320244853f0eddd315824d3 GIT binary patch literal 1507 zcmV<91swW`P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/gtk-floppy.png b/Media/Themes/Umami/Icon/devices/gtk-floppy.png new file mode 100644 index 0000000000000000000000000000000000000000..dc1d789207c3f881b2511423e3260557db9e2bea GIT binary patch literal 980 zcmV;_11tQAP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 DGb|cb literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/joystick.png b/Media/Themes/Umami/Icon/devices/joystick.png new file mode 100644 index 0000000000000000000000000000000000000000..c56ea7212dc482204bbdf1f53cdd3bdbe365d9b3 GIT binary patch literal 1065 zcmV+^1lIeBP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/kxkb.png b/Media/Themes/Umami/Icon/devices/kxkb.png new file mode 100644 index 0000000000000000000000000000000000000000..f9b6326b04d5d58ee3c879722fa249bef8eb3c78 GIT binary patch literal 890 zcmV-=1BLvFP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00LJ@L_t(Y$L*8POH@%5 z$G_*Cd*94^<4Hsk9ju-*LCB66UcR4Rd#i* z>#~SBDWZy6#}-9IlTBI|bC`{pp}N&Fs(F{ zC@?dEFn}=z#uQL6CQ>won7^9G^xNsX03HA^C3n!+c&sfNjWQ|1T45D=e;G^5S!8ny z$ara_GvD!J{u|~O(#Zbwkj*V(Ztf#MKqL~ud^%$%lf7+QC$qH!5>Z#0nwoXZ;aU`p z!OUz5kw_H83?f=L5KOUv)|S%%0MGN_*cD)A)YaFqQxU%m;C0Caw&NtcMNcLs#z91= ztgJ#|tpJ1}K(u}Wh*!t)V)i)zprPS7CSOj18Hh%0sXkP74ZuAB;kF&r*B@!`xYz+p zTYpqfh!EjMG6JEs004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00iYpL_t(Y$K96QPg8do z$8|%uv6wAevX@(vabP3#{XyR{mPMmKW#7bum>#0GNt93qi$Wn$t%j@E?!u@D6e zQsP89ovEbK8Sq{KMza+L(*kUa9UR9Y7L}0Cts|Sw;>M5H)+lE9ZzrF8Jh@sdmBI0g zP^vVTRj8p*KZnj}fz9rO!{LBW;1L%6NUp?DE){Y6m$6cF%h8|y(w0k?zE4)Iuvi&P zKbgUk83p9C^LVP%L1(hV=5&LxGZ1(u0s;#u%z^OH>U;% zuatj#cn6ZO0E>1StW5{r?ZA>xKva@&ysfoV!=fV_^gxP);Nu-&t+Ozz?!sf4M0RJSHJ1IKbN`VNs62 zV;)Cu?(FAd|_|7HBbhBLvr~95-UOIpKHL_tY8>sb3T@r<7jJZEpHQ$$vXI0 zEQa;9ELLI>h=L0qhn{+%#UfQWr%>Shg@KKM3+GBo^;{(y(Ek25`|E4{tFAJ-J1J{%t!zFH|1OIXk4kxXY1jwRp? zMqywUp5bdZ1%8tptZYB`)v2Jv_tSBI{k_iWF^o35i>Cr-%v9zL2tp-{x? zdLD^X7Gg93D=WZYUxdlwh2F|Sqc=mP){sRE4X1>$v0LcwKAEH(?+|ajns_JiF7W~J z88t~h{KN33MY((dv6U3~B|7uGf733;#e?N|gu)THIX6{4Se@Q{Lqo%t6koMx_p2g1 zw_d(ad_sJ2`0zJF?Hz5EN54LTi*-W`ONcHf5nE0nBE=}N2e9N?yj|q9a}GVZ&mSWeSQ7jvuAs`o}Saisu%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ov JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/media-flash.png b/Media/Themes/Umami/Icon/devices/media-flash.png new file mode 100644 index 0000000000000000000000000000000000000000..90e5edbcf718e6ae3d41d0a3e279ddb8cf24d3ae GIT binary patch literal 987 zcmV<110?*3P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 JPDHLkV1kxcp+f)w literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/mouse.png b/Media/Themes/Umami/Icon/devices/mouse.png new file mode 100644 index 0000000000000000000000000000000000000000..0c72f7754a621ce1ed3907a602a2d99470834cac GIT binary patch literal 1288 zcmV+j1^4=iP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00at2L_t(Y$JLepYuscU z$KTJBW%D!TA?V2Rf88Chf+0tALQIAw&xRVVdSv+qQG7&!$qTalM2>^!N8;VR0UFbJq|GOW=$l z`}?2o&0f2HY%HSo?Aa?KkqA)~1!H4l zkR%Dsx(Qj5@!G4yD3wb1@rMf|SFg^!5|77|wrx)VI8v|G5RmBZKKb~QdvsmbA%uXU zC;$M@^WghFIOkxD!EqdPCwfRUy2SZ^u(-I0`T2QRmIWaM zf*^np0*o=kxiU>jH{}{9}1c0vrl#2yemPK9HT_HpW0U=}!pbGIk50y#Nl+uQPAPB%1gYWzBJP)4dAsUT>bH1XtD*&8x6bc2D%VjXeAj>i+r8P+X1l5>A}bOnIda7k;6mVS~j(QH!v{J>H9ur zXJt>Qc;JN{s;Z6c-u>vVL~pNAU9NQi z;05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hi yZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgH7hVMIxsLI{$eTs0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Y}eL_t(Y$JLVCYg|YlDUglNfU`oyJMhq?0q3oOAZ}VUk0V$+SiB!3BG>SZnR|?Qg9Q{_iFL z_U?P?jLYRZ@Nb*hwteRFPrrC}b<7Rma=8xde|kT~V!=tMTXj+^`O>Xv8uK@9aQ?!D zXMl150Hq|EOor*{X(y>^+J^l#O+(W(48ve+^HwxX!|&Hkz_Y&A0oYVmNBa6Si3jf{ z7S+)-4{nbSO$cn;!jdMYSwJZhTbf8&tR+m-L`sPe0;LqTZ9C1Rlqj{V9i`URStSNuulL)E0HIKrNF+jg zTRTb!K!OtVU+O2(k>JumKOnFylgxAmpYF%)c4K4>eE0ZiXlMZVd1&y8sJP@ifiQmS+8XM!z+s4K?Kz73& zNGVBl>|%0!ijMXKqrZ=$lwxLP2Gcb0csz7>b#vnQNlv_XlEYnx0Vo!W{5mv5ZEY<; z4=_}bL0{iFLg6ZUdOo2cUXN0WTrLm53rCL9vafnYlNV`jZlE$>juc@^B?u~oqg_hO8M>TRd%+vkxX8stFs$`(;u7yAe~N=%`SfY{;3ap zR=*MJ!2Cc6hypPn0Qlo)ItAn|G-5QrZ-c!=8CO?>^$R{#tS{xUY4NJTZ#~yz)8V-kE5kfoxGy+q=m%vNs&VIIV2Yvn<`x_j`(aGMtWNrWe03~!q zSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuX zVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*PeNFgh?WBK~4300000NkvXXu0mjf DGb|cb literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/network-wired.png b/Media/Themes/Umami/Icon/devices/network-wired.png new file mode 100644 index 0000000000000000000000000000000000000000..111fe960f132fe01f6d1d0a52b569d9fdbc1f240 GIT binary patch literal 1324 zcmV+{1=IS8P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00b^cL_t(Y$JLe3Ya?YG z#-AjUOs3W}X)vJ@K7vu5x{OIVYxdYNJ4B$BKhnpX~ zKK;rS8qiD7**8I`e%Nd_OiWB*W@d(IG#b~F$>fK@V6eq;++9%=TmKpm3WdJD@!DG! zug8V9<1&cN2LPy4Dp1uToX$Q>O-*5Xdip6;SpYAb2e8%a^-q_UmWZZl za5|k>UtdQilL1ZB@Oa!vBoZi*) z!^6WCmSqu-$D!*wSeAvu;eetj$Y!%}yFFmphuYb>3q?^r7e!HdEC2wVOeT}Py}kDm ziNx%{z<@0f2q2TmKv5L<{eCbEgIH__+uPfqC<^g-96T@7i^b9>g1|S=16Tprvf1ok z7K>#wpU=OG}HtyX8p#zN1G zjfGZue(w~3>XH?JwOA}R3WY+n-EKdpC`$k24C(V`#_^ zkH>?2KL0!+#LDyhkEbqLXZ(n2wOT8e%iW=AIw(n!FP%<96vbUhy8ngC)jwUU)o7OW z!0YwG=ktN0tm=aYd0Qwn{wL4xWxD~!O*D7z#IZC@)04?$lwlaNq9~b-jg2p5S!#V(&-&(U1|c* zQmNEEzkkS5tyWU4mi~LWT>5r-d08+FPdzwYr>X~l9RS^tt#b&BHj++i?b4-7j={k} zcQhLP6+j(8#aQTB0AoLxE^`B5>H5w{XC!Lk;cw&kVR{^(8)T#~j05UK#FfA}S zEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tX iR53C-GB-LgH7hVMIxsLI{$eTs0000 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/network-wireless.png b/Media/Themes/Umami/Icon/devices/network-wireless.png new file mode 100644 index 0000000000000000000000000000000000000000..63c46e7b297ca2a9b82e11d70ef0f181edc0cd71 GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Uu3L_t(Y$Gw%$OI%kJ z$3N%Xd86Zu6O9rS#xOfEaToKG5VB~BLjHpag&222TP%S>3GG4GpY%}$w_O~z0&wsB_O?czO$slOH40~hW+_rCMF-}}DrKKPGAY6qiZ zBR#-pK>so8&8X^+cfR~;e#gw{*vPrWufNz_E^Ch)xm?yhc=(eYzxUloTXh39jL*;a zT)=fLtE;P?JQ1&DvzVnN^yN!*c^S8}6Zh;{D&5`K=4Pz509b43{kV^Z(+@rck^ou* z=x=Ll$8}vq1XV>#C7fiE;M5f6*)v2H_u@r#Z4Kw~V}j%31k!2L--xQx*49q#5eK#( zm8yD+i$zQ_2_HV--oA~E#SrmcSp~4Ji#~sjNu|)YZz=coVnw!>+X)a61kmMWXm7_} zx`Y_xx#8VAq)@<~I)y|caODc_nKS52B<2!wuTym?dd@S0eJr&O(x;R z3n&y2W3ZPmV{hKXHaEl7tLV}awxi>a3Q8r2$I(y-%4H;-MvQ@5w;&dSSFez{Ippb6 zY+oNX7K36DQ!E}5(9nQ{!(M-V9cgStZrlJ7D3zeQ8&03bCK6C6KuZg@wH3S$dhg1*O5rdi zzzKyw1Z`=7PzYB990%&^4pvb8xVF6QIA}PGitt-F%s1DslM98=%}wOwNhBKeHWLU` zZM-jFH;9Bnh$??%GW2U80R{4g;=T_l3~CeWh-_JXBS zDFz1z$z(E2PfxS3uuv_ySMhxnY|qWk&N4YU$;!$Kb8~YflgWJx@9CoLVXb9jV}sV# zR)&X%$!4=85(yR;7bz491cO29>+25%n91jJ{n6+N%H{IE<0Kgok{hd@42vbr6UJ9p4>SOcK-eEUfD8N$tOI`n8^B*c*`MPe0C)}) zALjT#%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q lEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1jFW+%^CJ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/nm-adhoc.png b/Media/Themes/Umami/Icon/devices/nm-adhoc.png new file mode 100644 index 0000000000000000000000000000000000000000..63c46e7b297ca2a9b82e11d70ef0f181edc0cd71 GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Uu3L_t(Y$Gw%$OI%kJ z$3N%Xd86Zu6O9rS#xOfEaToKG5VB~BLjHpag&222TP%S>3GG4GpY%}$w_O~z0&wsB_O?czO$slOH40~hW+_rCMF-}}DrKKPGAY6qiZ zBR#-pK>so8&8X^+cfR~;e#gw{*vPrWufNz_E^Ch)xm?yhc=(eYzxUloTXh39jL*;a zT)=fLtE;P?JQ1&DvzVnN^yN!*c^S8}6Zh;{D&5`K=4Pz509b43{kV^Z(+@rck^ou* z=x=Ll$8}vq1XV>#C7fiE;M5f6*)v2H_u@r#Z4Kw~V}j%31k!2L--xQx*49q#5eK#( zm8yD+i$zQ_2_HV--oA~E#SrmcSp~4Ji#~sjNu|)YZz=coVnw!>+X)a61kmMWXm7_} zx`Y_xx#8VAq)@<~I)y|caODc_nKS52B<2!wuTym?dd@S0eJr&O(x;R z3n&y2W3ZPmV{hKXHaEl7tLV}awxi>a3Q8r2$I(y-%4H;-MvQ@5w;&dSSFez{Ippb6 zY+oNX7K36DQ!E}5(9nQ{!(M-V9cgStZrlJ7D3zeQ8&03bCK6C6KuZg@wH3S$dhg1*O5rdi zzzKyw1Z`=7PzYB990%&^4pvb8xVF6QIA}PGitt-F%s1DslM98=%}wOwNhBKeHWLU` zZM-jFH;9Bnh$??%GW2U80R{4g;=T_l3~CeWh-_JXBS zDFz1z$z(E2PfxS3uuv_ySMhxnY|qWk&N4YU$;!$Kb8~YflgWJx@9CoLVXb9jV}sV# zR)&X%$!4=85(yR;7bz491cO29>+25%n91jJ{n6+N%H{IE<0Kgok{hd@42vbr6UJ9p4>SOcK-eEUfD8N$tOI`n8^B*c*`MPe0C)}) zALjT#%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q lEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1jFW+%^CJ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/nm-device-wired.png b/Media/Themes/Umami/Icon/devices/nm-device-wired.png new file mode 100644 index 0000000000000000000000000000000000000000..111fe960f132fe01f6d1d0a52b569d9fdbc1f240 GIT binary patch literal 1324 zcmV+{1=IS8P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00b^cL_t(Y$JLe3Ya?YG z#-AjUOs3W}X)vJ@K7vu5x{OIVYxdYNJ4B$BKhnpX~ zKK;rS8qiD7**8I`e%Nd_OiWB*W@d(IG#b~F$>fK@V6eq;++9%=TmKpm3WdJD@!DG! zug8V9<1&cN2LPy4Dp1uToX$Q>O-*5Xdip6;SpYAb2e8%a^-q_UmWZZl za5|k>UtdQilL1ZB@Oa!vBoZi*) z!^6WCmSqu-$D!*wSeAvu;eetj$Y!%}yFFmphuYb>3q?^r7e!HdEC2wVOeT}Py}kDm ziNx%{z<@0f2q2TmKv5L<{eCbEgIH__+uPfqC<^g-96T@7i^b9>g1|S=16Tprvf1ok z7K>#wpU=OG}HtyX8p#zN1G zjfGZue(w~3>XH?JwOA}R3WY+n-EKdpC`$k24C(V`#_^ zkH>?2KL0!+#LDyhkEbqLXZ(n2wOT8e%iW=AIw(n!FP%<96vbUhy8ngC)jwUU)o7OW z!0YwG=ktN0tm=aYd0Qwn{wL4xWxD~!O*D7z#IZC@)04?$lwlaNq9~b-jg2p5S!#V(&-&(U1|c* zQmNEEzkkS5tyWU4mi~LWT>5r-d08+FPdzwYr>X~l9RS^tt#b&BHj++i?b4-7j={k} zcQhLP6+j(8#aQTB0AoLxE^`B5>H5w{XC!Lk;cw&kVR{^(8)T#~j05UK#FfA}S zEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tX iR53C-GB-LgH7hVMIxsLI{$eTs0000 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/nm-device-wireless.png b/Media/Themes/Umami/Icon/devices/nm-device-wireless.png new file mode 100644 index 0000000000000000000000000000000000000000..63c46e7b297ca2a9b82e11d70ef0f181edc0cd71 GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Uu3L_t(Y$Gw%$OI%kJ z$3N%Xd86Zu6O9rS#xOfEaToKG5VB~BLjHpag&222TP%S>3GG4GpY%}$w_O~z0&wsB_O?czO$slOH40~hW+_rCMF-}}DrKKPGAY6qiZ zBR#-pK>so8&8X^+cfR~;e#gw{*vPrWufNz_E^Ch)xm?yhc=(eYzxUloTXh39jL*;a zT)=fLtE;P?JQ1&DvzVnN^yN!*c^S8}6Zh;{D&5`K=4Pz509b43{kV^Z(+@rck^ou* z=x=Ll$8}vq1XV>#C7fiE;M5f6*)v2H_u@r#Z4Kw~V}j%31k!2L--xQx*49q#5eK#( zm8yD+i$zQ_2_HV--oA~E#SrmcSp~4Ji#~sjNu|)YZz=coVnw!>+X)a61kmMWXm7_} zx`Y_xx#8VAq)@<~I)y|caODc_nKS52B<2!wuTym?dd@S0eJr&O(x;R z3n&y2W3ZPmV{hKXHaEl7tLV}awxi>a3Q8r2$I(y-%4H;-MvQ@5w;&dSSFez{Ippb6 zY+oNX7K36DQ!E}5(9nQ{!(M-V9cgStZrlJ7D3zeQ8&03bCK6C6KuZg@wH3S$dhg1*O5rdi zzzKyw1Z`=7PzYB990%&^4pvb8xVF6QIA}PGitt-F%s1DslM98=%}wOwNhBKeHWLU` zZM-jFH;9Bnh$??%GW2U80R{4g;=T_l3~CeWh-_JXBS zDFz1z$z(E2PfxS3uuv_ySMhxnY|qWk&N4YU$;!$Kb8~YflgWJx@9CoLVXb9jV}sV# zR)&X%$!4=85(yR;7bz491cO29>+25%n91jJ{n6+N%H{IE<0Kgok{hd@42vbr6UJ9p4>SOcK-eEUfD8N$tOI`n8^B*c*`MPe0C)}) zALjT#%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q lEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1jFW+%^CJ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/printer-remote.png b/Media/Themes/Umami/Icon/devices/printer-remote.png new file mode 100644 index 0000000000000000000000000000000000000000..bee3c98ac44f4401ebaa346e8b681aa9d1fafbc4 GIT binary patch literal 979 zcmV;^11$WBP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00PxXL_t(Y$JLd;ZxmM$ z$3O2G|AJFwA+XKbMT~MT+&aP_BqAyh6%aa--uw-xJ1KUego2thK?qWUZ8;JOiee0M z;eyU3KO%u0yyxQ1zWDZTci)>7x8B>ucMdk9$X6P@r`?(P&dhr=5B~2^?S9P8%{&0U z{y;wuuUx)*ZC}i@b2B0$VvN~SL_}uiW~7~;L;ppzy1GiSSZt=OR4k6;FfcHXtcL8H z0HD9WAEgvp>s@OC8ZZYU*x1-WDV53qV0d`=;|Y{f3=R&aB2VrU2wK_eRv+_)+<2^@ zHLmO8x-RSM>u9Z8mMCenCqcfD`v!QP%)oIR9LHg3Xef5PBMCf@Y*kx=Na6m}ci#h0 zsZ;>4)?$ny2m*ZHr&g=sc^=!_+p*4QHVbe+U&vig`cjqJelJlQW?*A>@wWo2bW@`YU752;(USuU~o z^cfQqr#N=(ILph+jE#*UB5}QqF>zL61*MxMzjyC0LFi+QZ8{Z@XcCCn$a5o$ zuAVN8vGn%#5{7#al*?sCN3#qL^i$p}qXCKTfrt=<0smAgRI47kV`nBcN~g7^yQ_28 zB}fs3VTd4PGDqm{>84hzfq=1%h^;a7^&Le-5Noi~$WTPE)*_;)*Xzj^ut|cS|6KTE z^5TU{{Qme62f@jaFL?F(1@(IU(Jlf%2<~)sc1~_>mHF!2`Bu{sV?vs(BdHzPz0(pMz&2lYX=Q<0G-j=6YX~hOcbRTt(F!q+9$`qP(q%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1m1d Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/printer.png b/Media/Themes/Umami/Icon/devices/printer.png new file mode 100644 index 0000000000000000000000000000000000000000..bee3c98ac44f4401ebaa346e8b681aa9d1fafbc4 GIT binary patch literal 979 zcmV;^11$WBP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00PxXL_t(Y$JLd;ZxmM$ z$3O2G|AJFwA+XKbMT~MT+&aP_BqAyh6%aa--uw-xJ1KUego2thK?qWUZ8;JOiee0M z;eyU3KO%u0yyxQ1zWDZTci)>7x8B>ucMdk9$X6P@r`?(P&dhr=5B~2^?S9P8%{&0U z{y;wuuUx)*ZC}i@b2B0$VvN~SL_}uiW~7~;L;ppzy1GiSSZt=OR4k6;FfcHXtcL8H z0HD9WAEgvp>s@OC8ZZYU*x1-WDV53qV0d`=;|Y{f3=R&aB2VrU2wK_eRv+_)+<2^@ zHLmO8x-RSM>u9Z8mMCenCqcfD`v!QP%)oIR9LHg3Xef5PBMCf@Y*kx=Na6m}ci#h0 zsZ;>4)?$ny2m*ZHr&g=sc^=!_+p*4QHVbe+U&vig`cjqJelJlQW?*A>@wWo2bW@`YU752;(USuU~o z^cfQqr#N=(ILph+jE#*UB5}QqF>zL61*MxMzjyC0LFi+QZ8{Z@XcCCn$a5o$ zuAVN8vGn%#5{7#al*?sCN3#qL^i$p}qXCKTfrt=<0smAgRI47kV`nBcN~g7^yQ_28 zB}fs3VTd4PGDqm{>84hzfq=1%h^;a7^&Le-5Noi~$WTPE)*_;)*Xzj^ut|cS|6KTE z^5TU{{Qme62f@jaFL?F(1@(IU(Jlf%2<~)sc1~_>mHF!2`Bu{sV?vs(BdHzPz0(pMz&2lYX=Q<0G-j=6YX~hOcbRTt(F!q+9$`qP(q%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1m1d Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/printer1.png b/Media/Themes/Umami/Icon/devices/printer1.png new file mode 100644 index 0000000000000000000000000000000000000000..bee3c98ac44f4401ebaa346e8b681aa9d1fafbc4 GIT binary patch literal 979 zcmV;^11$WBP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00PxXL_t(Y$JLd;ZxmM$ z$3O2G|AJFwA+XKbMT~MT+&aP_BqAyh6%aa--uw-xJ1KUego2thK?qWUZ8;JOiee0M z;eyU3KO%u0yyxQ1zWDZTci)>7x8B>ucMdk9$X6P@r`?(P&dhr=5B~2^?S9P8%{&0U z{y;wuuUx)*ZC}i@b2B0$VvN~SL_}uiW~7~;L;ppzy1GiSSZt=OR4k6;FfcHXtcL8H z0HD9WAEgvp>s@OC8ZZYU*x1-WDV53qV0d`=;|Y{f3=R&aB2VrU2wK_eRv+_)+<2^@ zHLmO8x-RSM>u9Z8mMCenCqcfD`v!QP%)oIR9LHg3Xef5PBMCf@Y*kx=Na6m}ci#h0 zsZ;>4)?$ny2m*ZHr&g=sc^=!_+p*4QHVbe+U&vig`cjqJelJlQW?*A>@wWo2bW@`YU752;(USuU~o z^cfQqr#N=(ILph+jE#*UB5}QqF>zL61*MxMzjyC0LFi+QZ8{Z@XcCCn$a5o$ zuAVN8vGn%#5{7#al*?sCN3#qL^i$p}qXCKTfrt=<0smAgRI47kV`nBcN~g7^yQ_28 zB}fs3VTd4PGDqm{>84hzfq=1%h^;a7^&Le-5Noi~$WTPE)*_;)*Xzj^ut|cS|6KTE z^5TU{{Qme62f@jaFL?F(1@(IU(Jlf%2<~)sc1~_>mHF!2`Bu{sV?vs(BdHzPz0(pMz&2lYX=Q<0G-j=6YX~hOcbRTt(F!q+9$`qP(q%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1m1d Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/printmgr.png b/Media/Themes/Umami/Icon/devices/printmgr.png new file mode 100644 index 0000000000000000000000000000000000000000..bee3c98ac44f4401ebaa346e8b681aa9d1fafbc4 GIT binary patch literal 979 zcmV;^11$WBP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00PxXL_t(Y$JLd;ZxmM$ z$3O2G|AJFwA+XKbMT~MT+&aP_BqAyh6%aa--uw-xJ1KUego2thK?qWUZ8;JOiee0M z;eyU3KO%u0yyxQ1zWDZTci)>7x8B>ucMdk9$X6P@r`?(P&dhr=5B~2^?S9P8%{&0U z{y;wuuUx)*ZC}i@b2B0$VvN~SL_}uiW~7~;L;ppzy1GiSSZt=OR4k6;FfcHXtcL8H z0HD9WAEgvp>s@OC8ZZYU*x1-WDV53qV0d`=;|Y{f3=R&aB2VrU2wK_eRv+_)+<2^@ zHLmO8x-RSM>u9Z8mMCenCqcfD`v!QP%)oIR9LHg3Xef5PBMCf@Y*kx=Na6m}ci#h0 zsZ;>4)?$ny2m*ZHr&g=sc^=!_+p*4QHVbe+U&vig`cjqJelJlQW?*A>@wWo2bW@`YU752;(USuU~o z^cfQqr#N=(ILph+jE#*UB5}QqF>zL61*MxMzjyC0LFi+QZ8{Z@XcCCn$a5o$ zuAVN8vGn%#5{7#al*?sCN3#qL^i$p}qXCKTfrt=<0smAgRI47kV`nBcN~g7^yQ_28 zB}fs3VTd4PGDqm{>84hzfq=1%h^;a7^&Le-5Noi~$WTPE)*_;)*Xzj^ut|cS|6KTE z^5TU{{Qme62f@jaFL?F(1@(IU(Jlf%2<~)sc1~_>mHF!2`Bu{sV?vs(BdHzPz0(pMz&2lYX=Q<0G-j=6YX~hOcbRTt(F!q+9$`qP(q%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1m1d Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/stock_mic.png b/Media/Themes/Umami/Icon/devices/stock_mic.png new file mode 100644 index 0000000000000000000000000000000000000000..58f1a80c27f384d1cbb6ca6fce0de39e9feda719 GIT binary patch literal 1088 zcmZXTc}&v>9L2vK%mEG4DMZl02(gG{hx+BXdg>mZ!B!msdO#ulbnCOgIn8}i5FYo>J@_GNga!y33rNtQw z0I*~-Xi)%Qa}+ZZ0GQKGi~@kEoElCA_^KNFO8hneI7k^$;Q+Z#0EMLhF9D!%6Ce{0 zU=#r$^#G^SKYJWW0RZ!K#zz?dmUBnJpw5li0GLS*n@5NLi7(*t1zZ47sZ{@#tne^e z2r81q0f1hwHy8{6U~g~F7jPNOkYGBB(x`#d0KR}55J;f}kjWIkhK2^QRKyo>FMS+C z@h88T4Wb5yu^D^;H(J0CV>9^zE{Ah5UL4C}Guh!Rkt8mP8_5@NRXG{ze}o33!NI{= zT^oWRC`v_9G$e!`8X6MH2w^fAEEbE!W=2GWb0Q<6ct>D#3@_$Vv_vYACP-xRL|IaT zEGbcbWLc6zu1HQ&q{tL0^5o>?Pf}CT(o?QvT)vudB~z7=m7STLot2lDn}03$+VyJ% z1=ov;3X6)0ic1U1%F1q3lvP%hSKh3ws;a84u2!qn_4Rifo9dfd8k;put*xzJ+-ufo zG;M8dZJnC-_V$j>cCA*c)wSt#_jNj5cXwA$Pfu@e?}NUc{{H@ffq_B&;P8lHbZm5N zY;=5ld}4B9a%ysVdV2cF)XdDx{QUgF!ouR>;?mO6^78V^%F62M>e|}c`uh6D#>VF6 z=GNAh(P-S>-rm{S+1uOO-`_ttI5<2!tTrh>3jn^#s0c1#-ZC|}IDXQ~+V;%Z_djsJ z;cyo|B>9plfk9N179&j0$}TFY{j{;UqvOH%h9`@wYrA_dUcP$$`;Kis0AM7nP+Hv| zCX4+5mQ8x5V zJ{F5*y~8tK+j93mb^f{6Y(lAKtJBmtVzSjcG))lydG7~9gU!YBWJ{%8S#;2n&`Ml! zu9f(8SJrarv*!k{IkQktFT1)7w?bqg^~*jr&WZi`(^*eNR6$>*f5omx*Qw-W7uV4W zuCSPc`>ye?+{fP{#@~d?k+`Z@k%JxoCp~@G%&pi%{p=*C7G~eW){5jMFBSs*U3y)9 z9`tdW9jb}sRm7iYr(-+4TO2%H$q)6y_{cajr^(UXTAQ8I%PZDK>s?33Ve^M2w^_g7 zNF}^Sl89uxvjR^ke;ha10^DK@Uzq&0|12klp}}uO%)=-(=DwXx*r&_-`y5xpM#j literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/stock_printers.png b/Media/Themes/Umami/Icon/devices/stock_printers.png new file mode 100644 index 0000000000000000000000000000000000000000..bee3c98ac44f4401ebaa346e8b681aa9d1fafbc4 GIT binary patch literal 979 zcmV;^11$WBP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00PxXL_t(Y$JLd;ZxmM$ z$3O2G|AJFwA+XKbMT~MT+&aP_BqAyh6%aa--uw-xJ1KUego2thK?qWUZ8;JOiee0M z;eyU3KO%u0yyxQ1zWDZTci)>7x8B>ucMdk9$X6P@r`?(P&dhr=5B~2^?S9P8%{&0U z{y;wuuUx)*ZC}i@b2B0$VvN~SL_}uiW~7~;L;ppzy1GiSSZt=OR4k6;FfcHXtcL8H z0HD9WAEgvp>s@OC8ZZYU*x1-WDV53qV0d`=;|Y{f3=R&aB2VrU2wK_eRv+_)+<2^@ zHLmO8x-RSM>u9Z8mMCenCqcfD`v!QP%)oIR9LHg3Xef5PBMCf@Y*kx=Na6m}ci#h0 zsZ;>4)?$ny2m*ZHr&g=sc^=!_+p*4QHVbe+U&vig`cjqJelJlQW?*A>@wWo2bW@`YU752;(USuU~o z^cfQqr#N=(ILph+jE#*UB5}QqF>zL61*MxMzjyC0LFi+QZ8{Z@XcCCn$a5o$ zuAVN8vGn%#5{7#al*?sCN3#qL^i$p}qXCKTfrt=<0smAgRI47kV`nBcN~g7^yQ_28 zB}fs3VTd4PGDqm{>84hzfq=1%h^;a7^&Le-5Noi~$WTPE)*_;)*Xzj^ut|cS|6KTE z^5TU{{Qme62f@jaFL?F(1@(IU(Jlf%2<~)sc1~_>mHF!2`Bu{sV?vs(BdHzPz0(pMz&2lYX=Q<0G-j=6YX~hOcbRTt(F!q+9$`qP(q%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1m1d Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/system-floppy.png b/Media/Themes/Umami/Icon/devices/system-floppy.png new file mode 100644 index 0000000000000000000000000000000000000000..dc1d789207c3f881b2511423e3260557db9e2bea GIT binary patch literal 980 zcmV;_11tQAP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 FV1kwbK)L_` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/video-display.png b/Media/Themes/Umami/Icon/devices/video-display.png new file mode 100644 index 0000000000000000000000000000000000000000..5d5ac83f7b903118b6234c0178f0bcd99c66c03a GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00TTpL_t(Y$JLb0Ya~?^ z#eeU;N_`|IJ(&r(5KJZ$BI?GC2<{v~7CMjg9xl30k+_c&*WBG%nR~S(JSBuJz`p{h}mB)s+Z=S(+r z748$3X>B}F_4JaJkFUhhl zdk^+mTwEG~8zB(!R03&MVaz$u1c{mvvZ53Ko_qe;iGGI&w5L}Mg^YKtL!e-c(yctL zM;(q^XUOpkBAy@b-eYO$F$xs$C~7TrLj#6$B~X+VN5`k6S&6CwhR_ChFRVVZf^+U- zf-KEYuXr^S!l=)6JZfj0wsTZ1K^PC}E(;J5e!BMy%gf8;c}|+9V+2`-st2`ny+Mb3 z7xA2>1tJDFl}z|-twlt*v2vX#ipa9;0tH!CTkR;-4t%S`)2+Y+5y9ll^*(bRk%|ZR z54f?iLYj6cigJ`7P1D|FTSu269~}gae~ii%ha^$&?7w2PUfJBrq0ujL&$g`YwyFIDE z82IGl7x7-k;0?*lG}c+vd$LYC1b@DGuNcf=f<_1`sfy-tGft8OXD3gyDh22f2SGHV zU^tr1CPh(LYi&95er;{-gQ}`N6p>3`*HIMR-rCxF_o4<2Fa;z@lFYsS=B>S(H(v=? zS66Y)p(to??!9^*{`QdV?QQvfXXn+upZE5F3`l{3-sQmf04BgJFjrO8?EKaF;G}i( zLi6O004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00PxXL_t(Y$JLd;ZxmM$ z$3O2G|AJFwA+XKbMT~MT+&aP_BqAyh6%aa--uw-xJ1KUego2thK?qWUZ8;JOiee0M z;eyU3KO%u0yyxQ1zWDZTci)>7x8B>ucMdk9$X6P@r`?(P&dhr=5B~2^?S9P8%{&0U z{y;wuuUx)*ZC}i@b2B0$VvN~SL_}uiW~7~;L;ppzy1GiSSZt=OR4k6;FfcHXtcL8H z0HD9WAEgvp>s@OC8ZZYU*x1-WDV53qV0d`=;|Y{f3=R&aB2VrU2wK_eRv+_)+<2^@ zHLmO8x-RSM>u9Z8mMCenCqcfD`v!QP%)oIR9LHg3Xef5PBMCf@Y*kx=Na6m}ci#h0 zsZ;>4)?$ny2m*ZHr&g=sc^=!_+p*4QHVbe+U&vig`cjqJelJlQW?*A>@wWo2bW@`YU752;(USuU~o z^cfQqr#N=(ILph+jE#*UB5}QqF>zL61*MxMzjyC0LFi+QZ8{Z@XcCCn$a5o$ zuAVN8vGn%#5{7#al*?sCN3#qL^i$p}qXCKTfrt=<0smAgRI47kV`nBcN~g7^yQ_28 zB}fs3VTd4PGDqm{>84hzfq=1%h^;a7^&Le-5Noi~$WTPE)*_;)*Xzj^ut|cS|6KTE z^5TU{{Qme62f@jaFL?F(1@(IU(Jlf%2<~)sc1~_>mHF!2`Bu{sV?vs(BdHzPz0(pMz&2lYX=Q<0G-j=6YX~hOcbRTt(F!q+9$`qP(q%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4 zWjbwdWNBu305UK!F)c7QEipG#F)}(bH##vjD=;uRFfbziVk!Ut002ovPDHLkV1m1d Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/xfce4-display.png b/Media/Themes/Umami/Icon/devices/xfce4-display.png new file mode 100644 index 0000000000000000000000000000000000000000..5d5ac83f7b903118b6234c0178f0bcd99c66c03a GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 Bp5Fif literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/devices/yast_soundcard.png b/Media/Themes/Umami/Icon/devices/yast_soundcard.png new file mode 100644 index 0000000000000000000000000000000000000000..625fd161ccc52a1d833b648e23e9b82fdf981ec7 GIT binary patch literal 1319 zcmV+?1=#wDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00b#XL_t(Y$JLeJZxmG& z$3OSZot@bq-PKqM+bPSE(AYG_7qC2P(D)Pl2aJgXUyKrX_5maM2SCsmqX{DZ0lfHX zqV0>?2;EX2pauf%Lfd6`ySts8+4#x4~<*5ixoj&$;^>D%BK1RQdM*hdZWe*4pj%zHByo3=<(l;(;GLwSFYqrrBZHka*}j9Jvof0_0{`5%d%eFvhC4@ zg$hz?=z0pJ6iO*HO-t6pu}-H$tJT7?tYBf`nmlyq;PRh;%r64SBo(DjzVh;s1B;ak zsZpHrwlS-u?al3+*-`-0p zQA#0%h@}w8I=)Y{*~GG}5-|EW4MeZkOMtcslv3n!IWn0HQ550(KD}O#AP7)O4Gvn@8Xq5L_wL=S ztgNuUzK-KKgkgvff+&gzf&kz5iK2)?p@3L*Zu9*eypgcz&UYWTjN zxM3K$uFKZe)<9e-MMR{>IC|^?i2Bcjn#&z9+m|++Z z0@rcuk!G{`8{h*j5JKOu13W-qT)bY{zrQ^4pkABlb{(XYf$O@PZqL=aU0dt*Tx@&G zvu(TWI9+YaZfm~psYatY_v59@p8_>t6^I@uGQcii^o=)PKQ=Qn^M0$f`2ssw7(~@h zr*r$_#UH)|>OeIH4|nRKZPS5gc{sFkaRZpKP9g^I04MI2H6A{r&M?pu-{c|DQh*VI z`#>)y={*I;e`o&!nDW7rw3yDj0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>% zI65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(b dH##vjD=;uRFfbziVk!Ut002ovPDHLkV1l)0VnYA` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emblems/emblem-favorite.png b/Media/Themes/Umami/Icon/emblems/emblem-favorite.png new file mode 100644 index 0000000000000000000000000000000000000000..0923492e3f3af918302b1fa03cda16ec830831df GIT binary patch literal 1280 zcmV+b1^@bqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00Z4gL_t(Y$L*6%Y*bYg z$A9;|H=Sua?G#&+mL?XFn1CQ!gGNfyqN0%mHY_wGMq=WI#Ke`7_*GbF+_;d18dk=I zA0cHyVxlI1bp_OFO00GSj*5#6bd)l1VEI_f(Rml5&;pM zh7_u)jkBM8qjeBH-3ucCUIpO7i<{f;b5s-%EiNn{6aZ@x5lo>_6&J9AwSp=7NG6Nh zq-k$XIg`_KFC!}eG0%CaZA(ksEG}Sujk5n#C?bF$LTPS=;PxWXodVHP09F9-Qkss3 zx7OU6o;wPh_!j_(d7g(Y6cJw|s+33pVoLx7qC$Y5&S8q-s@o@G!Hc1k$|GfHWihfc zn48J%ZEb3C#P>^8mn9M_poB16B$zFfdb5Vbnp;SxGh#$8RANr$!&td8JvFbLnmVi> zB7TVVBdi}{b%35+AedP{x$->Rj)%DZXIhJJb`?P2TyDOQot(HMPJJWR4>3ibU^b7Q zT*MSYHnQa7dl>E?)L|5T`5s(a1?UBBj2yW2_4Si`JL_6ptBT{Jt2a3gsjhCWj*JD9e@xy8tnBNB?5Y~pEAA9L;X3N4 zM;_hYxUHkr&0iiwj9G4IV^2`yuG$=)?`#A>sfA!Pp>EHT94t)j1(mlc6>kk|t0o0bq#DmcMisFmb=1tqX55Ew1 z!jPYTyyT~6=B|D&ype@G;8TL=^60MxP(a-pUK@wxam8EhiFj8$5$9&^?v+nPKJrV+ zi@;LVxI`%)tdz(B;y}#`j~l2NgpoF5yhZKLpB6p~Ap%U5M^#>|$hA@;7f6;RT9&B_ z=9bS05SFX-%hG6|04$c*HoQe{Ibj77E5|Aiv(nH?wH4hX{%iXi^A!ie4RrRe0000b zbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3Hnt zbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vkD=;uRFfbDqc((um02y>eSaefw qW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00ZbrL_t(Y$GwzWXdGn} z$A8~k_L|+zrp9Qumk?uHHL)cX%tL6j3L*$bDyY#4Q3`?&r4O~>gGk@R2Wcrt1*M9X zD2PaDOQ8=MrPdO}v=)*AfE;4 zxK#r503y1Hyy|D_(s?Hpzy8~W8FR<}v&cIFe(L4RrnX0(U*FlI$nozWm%4KgJpsgq zoy-gk$oTLtmpe~bJId2)`Jcg;d95zie6ab+y_y_33;C5*C*B}RQyU`GRPbe{3kf?9 z$!S7cI)#y#ZG36-lKsgQekj%{6>b!9B-Xu0b4Je>3d^?|ZNby?A)cP&s1H6{xeLca zPW?)(qg(as%DYFNUezrvF7>jnert!%XQo8SU3uh^rKL2I)%pAix|5;4wO!rbpd76N z*s!m@wOy49*J`%9LNu4G{Y-L}Xl#=R`?VLU0J_2hf#^Mge71IRE>${3(pauNaLdFS z2p}vwQoN;=dJF;88^kFd#kJu$NHc@*hLG{AII}|tRR`UV2yXyI2PBH7AuU_00u+Z; z!#m_s$jlt1aq`AD zkDlc1+X4VSy26=PV{;pmiKVUnwq2^vrz_G(t#K_t^P)r=F&ekX$&p*;+{DeVyZY$^ zDz>-wbFBaYAn?^Q?A;vZQ2qT6g~PG+o=`v)n#_T1VoO1G#gYrt^VZV%nDP4p@AaLd zAIJbJKnieb0X#s^g#>}%vEA%xiLg8BqurwrM1bjFEE?RrI?t(t1AGf)UGd_z4CHG8 zC<6olzx(S0ynycRQn~^rV7RWc?v-@!aXUb#V!}f8#Gs zr=V+lQ)D9m001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHYz>% literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emblems/emblem-noread.png b/Media/Themes/Umami/Icon/emblems/emblem-noread.png new file mode 100644 index 0000000000000000000000000000000000000000..4eb08fe7535e82a9c40a0de6ed483654eab77888 GIT binary patch literal 1060 zcmV+<1l#+GP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00RL@L_t(Y$JLchYa3S- zhMzkcNn`l~S2c9eg~=)q5)+bUQKE%6?I!vG?XrY%LU0lYPJV%RaUoT0jPb%=&w0;#W-ff{rrIy7 ztE&q5&bI9bTI=(JfF%-%znjhG0J(fbTL?4Tied(^Hb^9uL)LGP@dOcf^?dc>(a>O*x1;3ueJWtbzO-VXsyp> zvsp?jD_HN|4bQZDl*EqcOUWVaci{xI0G@Ji$(M8Tk5xFp?A zx{DkfV9I5*=SASB(!hws8h9QD-+T+o!gwCES`ch)@!{*MsALlH{pjPedbH@I$Rc6& zB8UihezRGP zIZ6*7VvdfYF%*$-Ts!_TA`*^gYtI@U=CgbEs8lKd2lMmuwXOvF`}>7rvH0`!^z_i= zwQB=iIOTGgVzC(P?Cd<--QE2H_$NXc85tR!ot>SToSeLwOeW9tg&qx{`FDGJ`{nxj z`jcw4`WH|-UZzcWQTT%-08*`I6=)2I(VzVX)o9hbc8$B90000bbVXQnWMOn=I%9HW zVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 z05UK!F)c7QEipG#F)}(bH##vkD=;uRFfbDqc((um02y>eSaefwW^{L9a%BK_cXuvn eZfkR6VQ^(GZ*pgw?mQX*0000 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emblems/emblem-nowrite.png b/Media/Themes/Umami/Icon/emblems/emblem-nowrite.png new file mode 100644 index 0000000000000000000000000000000000000000..967e62a69ce7e5c547ae62fc7d37579131790cf1 GIT binary patch literal 838 zcmV-M1G)T(P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 eGB-LgHY+eNIxsL37I?P+0000^mzaP literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emblems/emblem-system.png b/Media/Themes/Umami/Icon/emblems/emblem-system.png new file mode 100644 index 0000000000000000000000000000000000000000..77a84e841eb4ec3d2ce76171c0ca7a336c91719a GIT binary patch literal 1566 zcma)%c~FyQ5XP5MM^KrzI0Zzs2#N=y2^6r1S`-Q>P(>=$0p$>@wo+T7TyiNPU%n(H zkb`ijP-j{l6-7FgLQy~n5Ht$pgb)Y;azMDkeRcxswEe5ky!+1X&hGpCUgFUJZmQ5dZ3?(S}C7Ma1!re#sG88jw~{_<6mOeRB6 zd={O`U}kfh8hJb(f@Uz7SqxUTOeRyS)dB&Zoy*EEV9Vw5WoDF2AXA8KZIYgz9*ssr zXV6$Vbh%uvR4Ne^pGhS#m{~cw%!Y=Bc4@m>twvD%($@&=JXUTVE59IDC=>`qLXk)) z5{o1faYu)=udh#~QuX)u4-E7V4-XSb8BA7oVbQI}j~|zomX?*3Ra8_|R#sM3Rn^qg z)YjJ4)zxu0ocj9u#>Pf2w~@=`HZ?Wzc)T}nUN<*4H@CcLX=!O~ZEkICX>D!g^Z9%M zUmy_ZeT71iNF)}E#S*bZB9XL7+S}WuQfWs=hfLNXlgTr*O~DwRsDR%U3S$0sHxCMPGS zrlzK+r)OqnW@l&T=jRs|78Vy5Z3C0j0H6>T7;qd63|APhG5OeZt+|Dz)rO6>Tefbu z-|2vLa&~rcbNBE#c*w`s?`X*B@XO^D&tBBkaq78EFL^v(b8{*4f!5 z?^g8mE;H&0`vP(RUl|3PlS0$#{lUV}_IOppJza40{_+&%L`c$IGvoQOOM5@E z+8Z;{rd>H1-IW~cB~1Ng_#mZPx9_({A%ALCoVE0fI$;p1+qwEd1?Dcu=?cb*-G~=r&UELScM2|w;QWPUPZ{>+yXUg}P^m#a!7TTL% zwd!~prO5lVX=2yOKPoJD(6eI4YerhU_PRbeRr2J1xaNuac-W?hP;ArJlKVRDsH9Od zDr1HtMkOWRd}0>6?|B*LaG;;FN9H;^-Y<8z+dS-&UM-Ra_?`%%5lrl=8@LYEJ1(rd zZF+H&?XKE-4l-tw&ueD{Ifu3PP$Y!^HY;rp~l1uZ?Bp8~)ZoY%Lwv*EZXx5%HP z!f{dHh;hU?V7_p`xCCMMxH&nyVO{Jo7&i=trQ3u3*TSVA&t5ne4~`ft)&b+UrU#|eq;)7w={{}gFM{WQB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emblems/emblem-unreadable.png b/Media/Themes/Umami/Icon/emblems/emblem-unreadable.png new file mode 100644 index 0000000000000000000000000000000000000000..4eb08fe7535e82a9c40a0de6ed483654eab77888 GIT binary patch literal 1060 zcmV+<1l#+GP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00RL@L_t(Y$JLchYa3S- zhMzkcNn`l~S2c9eg~=)q5)+bUQKE%6?I!vG?XrY%LU0lYPJV%RaUoT0jPb%=&w0;#W-ff{rrIy7 ztE&q5&bI9bTI=(JfF%-%znjhG0J(fbTL?4Tied(^Hb^9uL)LGP@dOcf^?dc>(a>O*x1;3ueJWtbzO-VXsyp> zvsp?jD_HN|4bQZDl*EqcOUWVaci{xI0G@Ji$(M8Tk5xFp?A zx{DkfV9I5*=SASB(!hws8h9QD-+T+o!gwCES`ch)@!{*MsALlH{pjPedbH@I$Rc6& zB8UihezRGP zIZ6*7VvdfYF%*$-Ts!_TA`*^gYtI@U=CgbEs8lKd2lMmuwXOvF`}>7rvH0`!^z_i= zwQB=iIOTGgVzC(P?Cd<--QE2H_$NXc85tR!ot>SToSeLwOeW9tg&qx{`FDGJ`{nxj z`jcw4`WH|-UZzcWQTT%-08*`I6=)2I(VzVX)o9hbc8$B90000bbVXQnWMOn=I%9HW zVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 z05UK!F)c7QEipG#F)}(bH##vkD=;uRFfbDqc((um02y>eSaefwW^{L9a%BK_cXuvn eZfkR6VQ^(GZ*pgw?mQX*0000 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/face-angel.png b/Media/Themes/Umami/Icon/emotes/face-angel.png new file mode 100644 index 0000000000000000000000000000000000000000..50627bff5483841c0d05583ee3bfa958f7cd3af6 GIT binary patch literal 1496 zcmV;}1t004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/face-devilish.png b/Media/Themes/Umami/Icon/emotes/face-devilish.png new file mode 100644 index 0000000000000000000000000000000000000000..242560d14100c2c9d33606e2636688643061efdb GIT binary patch literal 1538 zcmV+d2L1VoP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/face-monkey.png b/Media/Themes/Umami/Icon/emotes/face-monkey.png new file mode 100644 index 0000000000000000000000000000000000000000..0c87d782bf2974a320e0834eb8512b322aecb4fe GIT binary patch literal 1411 zcmV-}1$_F6P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/face-sad.png b/Media/Themes/Umami/Icon/emotes/face-sad.png new file mode 100644 index 0000000000000000000000000000000000000000..e5f4d021da677344eafde3f61817215aa1314d2b GIT binary patch literal 1468 zcmV;t1w;CYP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00fyyL_t(Y$K{nvY?M_L z$A9Yw z_zW=^ib)$qc{Y^VQcB;YPo|wtr{Cj#_qrL9Qd`l5D^GHAb58F4|8nllJqP|brZu9m zvB7LS(qL{E?-R#4-9ljPdB`B{P&*4w9Lhl zo8s+{s&wM)noV2Qtbb@%1lcR1kpZ-9LwW63Od_Q(YPapR^JCE!J$(Z&KUG)MbLzry z+d?g4VWRQCu9!PH(YRsL=G>x{D=equBfdD@3b`Jlr~tEgIU-N<&WTebuAiW!rrLP^ zz@eB9Ssx#JX-m~T0V&%#zPhq19zikF@+RHGLmWEVgcFJ(vUAYUT*!&jd@;evrcrb- zf^9nNetcIn(CVurNAA|VEBNr$2e+7!SVPU$C&H<7@1a$SgYWm?xRO*lKo|zXFfdG$ z*|g-1589ZT9K-4Qmh$S2qO_!>THV;Re@TF&)UL|%O55-M1vN9mkF7q zvhTi&9N2vuH8F~sxP_j$g_<1Y>CI{CYg*V-n?eVQVDcu}`73b)tN8V2YH}8>zzwvT zof;+XOVE-aG0KMcEaprEO*YC&BP;`?BuIO#&YnXi(`ZkkC8)V+9AAp08&-<~0vVj| z>AJdaT_}#3x`p9NWZLD%m`D4_6m9(;t}>~u2(z)wqPkFExjxF15GrE)`URBqhd(>r zF}rvLzBqexaBxp)amdbhC8-&Y=1#~hE~TP;6GO;WK_pACHXOpPfYg+a zlo~4<;Y{mL+VR9UOSX!0ofBhQtMXQjBvR{(B6FbvTudPd8=5Q&vTcKi1(wzb-$y$x zmggZN2bZh>U?%AucC)Pd^J8YYifhUOmq=J5YztwTm&t>2^`5} zTJ!Zl+&?{Bkam3b|L|+y#f5-{?OJ4@k)on-@s6rxZ$(1p-ip{vXjPVHmSnj^OwGug z&2ZY*9n;y)j7xiKck;EqiBu<$1m=L??*Jhn0xSdaYs%u=i{s%Pxlyw&WSW%-VFfZ6 zn0A6|lS%hdTklkJMtBsM$^hh2fIGkY4kQPNW!7jWXJiB%U=~Pa0JDG#`17x6sgWr% zLgqZT=DnH9T*G`<8DDaI{wMY)HcmICQy_tr0000bbVXQnWMOn=I%9HWVRU5xGB7eQ zEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7Q zEipG#F)}(bH##vkD=;uRFfbDqc((um02y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(G WZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00fLlL_t(Y$K{qwY?Vb6 z$A9yE-+lJJTKcdQ(1r>EBIQ*Gm{1Un1c6n9i5lVpV?-lCWM{(0h$Ofo5n>b#irQ#Y zLLfp!5JiO+Bp|f3g_c_@(3ZBhy|>?MX53tpQhB*?<0L1OGnx7SGUw!+8Tj7}tr7M0 zb!PqMx}esGdp&p8jUR1%bg5|>D=bqjQkV~EAxumAm!vd)^tC?p+}p=K7oh(&VAuNE zR4C}Y7cU&MZ0fA((NuYbC>T?QHcjllZZg-d>Za3Yb6x3llke)fwcAfzzGuL$*A~pR z3=ULR&x}{iTN0E*jfg-mSUI#%2o(dF#44Ca`dp(reCTM-b)2;??`Zh!u7LWNYGZcT zZe9B1in3%$3b*qM%(4(>c{w7MKo|l&oJ00pM|BTirIz985_W&EBkK)&)f?VF)jCqo z7!4Q-+S{j0o|;M|5@b(q1uKh@OkyNbh=Mpent*r`Gnqt$HE!E3ET@ylm()h3R$p%3 ze6!|F&dzo7XPMzx-ISUqLfOW5!4iDiFv!#Ee&WEP4uoMK41=RTq*?vK5xzNk17Vri z%^x#i(hM=KqT)f-I%Cak0k%@Ll@lt1-j(C1!9OvyV%sNaWG+xi{0iM)z;w9tp+CJsf&LIZU8LUtjR;-y| zDoA)t7$w2EVdH09+#w%jE0Pm~%z4oxKDQi4T4kG7PL6Ng+dTLWfFrfa_Fp6E`Jg~9 zT14oXXBk&q%-B?lfbXNF#EM4g%jLLu?hv15AAEz@Bp^wi;NRex4eP$<;lw} zXID)NB$>3li!&b{C3@rtISsA1s24*()Dld6H35Mlx-T}Me6MTw!KUmjlk9ln`%9gj z?tlsqiAk#0#&8k_IwU9pCgyDnAj1Y;p-IigQX(lpglKNLfFsqfx6Og;`iHM{W&Eo# zD;SbNVnqRJ-k8Sq)4lYbAEd1%hZayYxqy=Cg*-C995vX99LS;~VZJ-_N6vP|*LSSs zS5{8io(MS4KUUTgt+cWP;xVkU2QU*QnDJ7CAyBy+$e|wmo_1t53p&ESbkslA8EDvZ zq+004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00fOmL_t(Y$K{q=Y?MV5 z$A9yE-+sH@?sm5~wik-2Dfbu(;bOqVifBXxUTl0q;sYH=q zXbfnCP(=(zXsd)4g@zx8412GvkA6QYtrJd~uSK$(hXjf0=VK=M4OBhSrFt zrbe@ASEH>p;%?8~apOlj9$05u#zxCj4=T(BS_sq9LAR8~Pl47)x4d!UQvv#41NObp zkPO@I`|;w^^^2C>7fn`Hi^9?r+B9*}BTNhr>GogGW(WHE+5=BFZrgL}@?8V=y|j9T zW$M~lX(4~W2VBGm6eEC5yBAYTo(V@FluxhyZCW@UC#aw_h$T@U;p9Ra8seV`jr7)M8oAf907o0r)6UMF7HJ}kGNN7gn( zrB+|;+I6Gm4bHpU?_Xv{VvP$|J{r!nz5|v(SQeih?&W+(7-1L)!@w|24j=6&Ef*jx z6Q|<vI;90X+fhDM3k4njqapN!=Q}K(D+dz?XV?yu1d#>mVo%fkZ2lt($A9tSUr}4kAJ(LWSAB zWf8gb1lo7dQlZAXiLa={l~(GNP4nZszwVg)2f&qDWyXd{_yK5%@_gjv7_U89f*CeY zBO?gQKxmVv8zKbhEJ4~wdjUECl^e(L12N%-)lF|819|%LrL&tChZ4ByL9(t#RcI2F zhboL;0K z-{jzrla!Xa6eV)ZDzPaqwrKD5xX_bhV8o}P&R|8kLKtB>E?vZx>X%#Q!1IFU-hqi= zC}!DVX;T{Z*{~?d*|8`m&gU5XBaP=P?yZiqd~S*jOQ6^vMr9;QM>uxoYS!_@SGRr1 z&u^Nyrzqq;^Kfc3T5Dyn;t`DE3apC7kgP-41>lb34YnbNE@HSo=rD(Rqrr*(P|L?Z zTv^Su;%ChSOip+^+yZOksbO=DPC^YXwm(2NeGT6@j#L^0fvpuG6%b@S+R~Edo|xlc zKE;g6YR&}w)-xvSYLd-kUUYMR&bFea6S9<~Kn566VgwR9@DPsASXS|6U&22!P?B|g zwjTQVO6yF3xJmIdfyk^`;nbSCvezRa^ND+6lcA~t&#aERL`=<4*5+!~)*a)8&ZNtQ zGhGueUm2b1049JOAa4f<0TG}KD4t)FSeZ(M*Azv~`jBbXB7`La**EUU^VcWbwsSpW zC-TC>z*rs-+^!!O(;#smmY<{fH6t(J02yE^56A#6;7<>exJ{=s^5Od1tZ6w@`7lg3 zl{fvL{ucfOTXZx#%J$nW0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vk zD=;uRFfbDqc((um02y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000< KMNUMnLSTX;ij{Ey literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/face-surprise.png b/Media/Themes/Umami/Icon/emotes/face-surprise.png new file mode 100644 index 0000000000000000000000000000000000000000..61427d67172a42503639a87790ddd3195c42014a GIT binary patch literal 1482 zcmV;*1vUDKP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00gH=L_t(Y$Gw$VY?VbA zhM)QW|153K(n8x~DU?)DHi3c!NHA=v1ftcehQuXA6K-6hA*dH>7ao#AXx*Zde3{IbnfFQFneQ9;kB9j$&fW`b z12oX}PXx+dokPB)6?R+x_;}f#%@6VCBUASR<0-v8sF5^EQ!uuu|6o%O^Cn{SXs1C z2o(Vt$0}MuPjizxdgQ09>o{9qXgK}pp8^g%w>Dyj?e^M@b%}UcF>deYn28W(MFk=f zLl^>`%OY6MlDsrJQNCIr|@t*x%A z4tm`uP^s$}T5+WDCdnHm;MyP+LTgYi`#x#q!qqsS(K3tE{w+&tY6=YL)=dcTJ$K!d z(t9j_unmnui68|*i?&PWAg2*d1mTuI4gv2}bp-MK^gz#^UAV)m3>!W>8xlwG- z7h_IH-4!5xdFr>ei;ph|#4*#?QRy^NNCcA3=KUPro5YI3=C_s+(kWCbgSI`C>qDTN z!Cz0Ke6Mfc;g-x@Epj~ZV`p!#JE{U`$4A*7D(B*lj&SF#lmDI1w~ zP+1r4YLpK1!})7j+ZA6<*sf2u-WYgrcJZ|2aQcC=a4ry+_8+CLHN7_lr>^<*Yt0k2 z6-+TxsGN(+xu8NEY|Z(X2GrS)k9EB~Uemg>l2c>ucBjCqYwS0xbTQ1(E@5A?X7zl! z<5Ng1Siqu{QF136kU1M=`!o$}j&?Xy+WJLPcse8<=;S8t?SfiWOAzHQ$P5CFnJDNr(J zMtpf99$FPEG;0E;S&a~u^kvVeEiVm?IcJ)?Mvmu&`+<=>z@N+_V;m$3MDlZCK4;_w zY#;-q^MDNC0N!L4jn5l-yKaRw?$xQh4dX-QL-T)*-vK%OL}@(cSt004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00fdrL_t(Y$K{q=Y?MV5 z$A9z9x8GjdZJ}jBXh}`w8bcvk3@B7YBZkWhjlLiu1|C3-5imRu!=v&li6|g35`zS9 z32KmVF^EQ>Tp}u!fA$bG*&fkyoirZp{2ym^pQ*V=!<86Ep&HwUi4kvx_aBmzaAK{d;Ofp zO@r^6nr2os&R$@tp*BQl0L%hf1c)F3Dv24NP3MKS;K-pL3a;aUbg_Wfnr6+v zo-rISWZB!MPM%g(Sy@T`)K)O_7|A3?WfdY`fsR!|MUu_$U1IZx9k`cwV>(?tyPzee zbny9>Eq7|(;e4>}iCHp|Xr0>pTsYtMK9~Ywn#ja-WVi`a&PN*c7*g`hp)N9N3c{4w z?R%)7G($`nKmL)RW5&vR0_-4YX{c|oyz4&^+`5UO1AND0$Ih>5Y1zca^>35QdJwqu zqzfE8d>XAaS`~0QzGY@pQ`AuIl5$P6-0?m4=}|RfO~3Cnnt(tB{M~P}c<~yRubM;A z>E(LIH%x!np`k8;Rtf?IfyU2VW>n)`TwgceGXhWP$0};;@cJ%+()4FMY7=3sn|m2J z`#Pd;CzEOgWQSO{Vj8b3n~3&&vqkDP?gvrY-5&TR6KjWcYAMHa>%wl`K$T{KQwx%ox%$r^_FK+26 zqs0)S*o$RJVu>hwj%GR2K19s$85`HAViA?ilF#Hxr;8Y_hiTQ4b)uX&*M4Ky!HZk( z9)Ty0Ug_#uG$9qTsvRF?dyJ{@Si8KLOE+_TaqLgF?eC>vL)_BLpB!Rgqk-l75DwFR z=?@$g{BqACyRLuadUww6NtjkxS;UGiS`4ADo}wW&olR?ALYQHK!FF`^8akIIDA;IM z6T~7MK7XTNyW*=6Z}MwPCvB??Ijf#Y^~V~_JZ97)h=y>4#1jTm2tp==(ct+6wujJB z4s^!+<6WWCpB}q5hqC5p4hP)Ixf`6Qx#VP8*6S*$(=K792!%|flpsKQfC^BakG6f< zGMXb-6Sj?fkzt)x4hNjOI;a}!s*Vi0v1MIF%Z!AX<*UxDt$rKG zIcF|h9XwtVP6LA_fIrfEV3a{BfJAAImDY@sfDPn z1*J5UTTn9npZ*s90d@u@YCnKa1ONa4C3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7z zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C- zGB-LgHY+eNIxsL37I?P+000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S*E^l&Yo9;Xs P00000NkvXXu0mjfYTSwy literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/stock_smiley-1.png b/Media/Themes/Umami/Icon/emotes/stock_smiley-1.png new file mode 100644 index 0000000000000000000000000000000000000000..7e57001250292736734545ac84e2fca6931d8062 GIT binary patch literal 1456 zcmV;h1yA~kP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00fOmL_t(Y$K{q=Y?MV5 z$A9yE-+sH@?sm5~wik-2Dfbu(;bOqVifBXxUTl0q;sYH=q zXbfnCP(=(zXsd)4g@zx8412GvkA6QYtrJd~uSK$(hXjf0=VK=M4OBhSrFt zrbe@ASEH>p;%?8~apOlj9$05u#zxCj4=T(BS_sq9LAR8~Pl47)x4d!UQvv#41NObp zkPO@I`|;w^^^2C>7fn`Hi^9?r+B9*}BTNhr>GogGW(WHE+5=BFZrgL}@?8V=y|j9T zW$M~lX(4~W2VBGm6eEC5yBAYTo(V@FluxhyZCW@UC#aw_h$T@U;p9Ra8seV`jr7)M8oAf907o0r)6UMF7HJ}kGNN7gn( zrB+|;+I6Gm4bHpU?_Xv{VvP$|J{r!nz5|v(SQeih?&W+(7-1L)!@w|24j=6&Ef*jx z6Q|<vI;90X+fhDM3k4njqapN!=Q}K(D+dz?XV?yu1d#>mVo%fkZ2lt($A9tSUr}4kAJ(LWSAB zWf8gb1lo7dQlZAXiLa={l~(GNP4nZszwVg)2f&qDWyXd{_yK5%@_gjv7_U89f*CeY zBO?gQKxmVv8zKbhEJ4~wdjUECl^e(L12N%-)lF|819|%LrL&tChZ4ByL9(t#RcI2F zhboL;0K z-{jzrla!Xa6eV)ZDzPaqwrKD5xX_bhV8o}P&R|8kLKtB>E?vZx>X%#Q!1IFU-hqi= zC}!DVX;T{Z*{~?d*|8`m&gU5XBaP=P?yZiqd~S*jOQ6^vMr9;QM>uxoYS!_@SGRr1 z&u^Nyrzqq;^Kfc3T5Dyn;t`DE3apC7kgP-41>lb34YnbNE@HSo=rD(Rqrr*(P|L?Z zTv^Su;%ChSOip+^+yZOksbO=DPC^YXwm(2NeGT6@j#L^0fvpuG6%b@S+R~Edo|xlc zKE;g6YR&}w)-xvSYLd-kUUYMR&bFea6S9<~Kn566VgwR9@DPsASXS|6U&22!P?B|g zwjTQVO6yF3xJmIdfyk^`;nbSCvezRa^ND+6lcA~t&#aERL`=<4*5+!~)*a)8&ZNtQ zGhGueUm2b1049JOAa4f<0TG}KD4t)FSeZ(M*Azv~`jBbXB7`La**EUU^VcWbwsSpW zC-TC>z*rs-+^!!O(;#smmY<{fH6t(J02yE^56A#6;7<>exJ{=s^5Od1tZ6w@`7lg3 zl{fvL{ucfOTXZx#%J$nW0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vk zD=;uRFfbDqc((um02y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000< KMNUMnLSTX;ij{Ey literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/stock_smiley-11.png b/Media/Themes/Umami/Icon/emotes/stock_smiley-11.png new file mode 100644 index 0000000000000000000000000000000000000000..04b6cf122032dea102ee96498972949ee0ef4b33 GIT binary patch literal 1490 zcmV;@1ugoCP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00gf|L_t(Y$K{nQuTPUTpP-shUyWQP$A0J$k1u5WzPkzal$v2t#|1$rXd;|ZRkP@+X z?{anT#^t({i2L1l&xPC9J@K@vDb{WJmO`kSblO~3sdc10yz1lQdj!aT4fty9 z!nj{Iw?*?uEqP?dW1;w%aUwjb7^$il1HB}=Z^-7;XVP7roz0FVm#^7UfBC)vyWU$o zN7uCjQ>Hu`nJ{gh&rLQUe0`v$k-|fG5pZK@;W>1kJ?R}ed@OC5=9)J@ZQOlNz~0vv zM%;XHY4J-di^BPNn5{cW>nZXkmLMW|2t^<>Y5Ff;BWk44@|R$HMeO);Ys${pm2ZD> zsx?cS{_aYQ(Z4CO&I#s}Nl7<7He^f~l^!!wJj zLay|_+qm&o&0C7k-k3W>4Mdh#Ot11M>o()`U!b<>CU0%n$I!5YP!xoskTi5^e(h%S zH&@UKr{X2fQ(7@il#Cnqkk>kG^@sq&^A?tsmg)ADI^4cilF1aWZMyi^Le~V>Mo61v zCWIqmZ2tNpXHPd^_xz68`XkdTD}#z_Rp)AIxrJj}RRyCT(45{zq>$8H&$nYRC#TUa$^s&2lK3C0Fy3<`Wo=kDJ!@~4b{P7~*{bV1{R?TPY!Af3^ zA4T{I>Ald1=h$644mGFl2rzANwC(EEB_+i^{lm^zD4ko)<)40GXmpSXr4yMsyM%(6 zpLE7xOHDnWZ@Jym^-hd<2bSd=>F7#0^=hGhU8~^z>RHqrtzmNT z8g}9u?ob20LJ$ZF)|dOxC&KVx8po7qp#VQOT}vC5*q5uR<`%!T-0YcI9v^!>=}qkr zQ#3;a=xJ+W*{Bl)H3;hp0Sz=M5w?vqOf<_vcz*V`WSnz7-szn+?Q2Io0YEZgt-e=?8J(yvZ(=; z1Ecg*{K>+;U`fy=-v^o|KnRc$&z1CBf`eUo_Nl(8VcM+zq5kTAz|Do+4iH(}0!2ms z;sxcSKMMHN7bZlKzOg||9TzkSsFE8QovySln}@Mg;hP z05BTJpEN!;yEx`wkQY)beX3f95Sr__okNCut}kJpI@>;YJS*G{3}yk&-7HdaAWK!iIxlNOZhF}q@%#Kw;!k%nAt+iKHHrWL z03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R) zMObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*YkOFgh?W6Bc;40000PbVXQn sQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@WXPfRk8UO$Q07*qoM6N<$f*M|+t^fc4 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/stock_smiley-13.png b/Media/Themes/Umami/Icon/emotes/stock_smiley-13.png new file mode 100644 index 0000000000000000000000000000000000000000..20b81486d1777d49c7f78f481fa9e8cf182e7c45 GIT binary patch literal 1383 zcmV-t1(^DYP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00e1CL_t(Y$K6(cOq6vP z{~iaz4OZgWKrA_3g(iw^MiUA`Y__(L)n*K2$PW=uFek!PbZfI(TXUA9xiqAIgd_=r zAnC~90F#6J1!%xQ!0{X8eeZbh2V(Qn^B(7a99e(*Zl67S-uHQ)&-eND-OJch|Fjjc zc+D}U?#N+t+WqX2b{~T%MZKNg8wyzb{|>G>%JljJ{+Mbw3fsWfa88$Cp8FQgsY?({ zUt+TU5XA4gJf{C|r#{WX#S@O#WKpnE*vE6w)e;MYOfxp)mqyPrrnNS&P9O!`oa^RkdSLA6g_WDtHC?xnbb->tHv13`3R- zeR&=lFHXf&!$%NC)4|>IbCp0CP{2MIj(PLzSQy)hhZDQ;U@R6+^K0~eBQ^VZSh`n& zYxjoG^&ZB{6VQ2>sh6((K{h806vNUNP6RTl{c{ld6?EDycrcxZhum5G%@tu`CLas; zPZGf<%$Z(*v$-TI&ma;hV7D+ebYLp$wh{*Cqx3n+!JH5$Gt+(gd_b^E8&TIxiyeg5}R=U>n|r zo(r+4^poIs#_+2fbzA(>McprwPEUo*a5Z zp)FM4`xOjSRJ0Om2_n%CL|7Z{jiIs);9GsbtH{2VGSK9B&OlFdZ0kM`OZ#Tb-eDNJ zMkjsMi-Ixy=*!T%cQ3Aad0`+a3Egi*qjI$b_1h&dU#8$cGgxn~1E)%+N%GJGV=hfm zbvA5WsoF@xm>gU}R*(UP~& zrSvdGeMeZlLEMA>VpyBjgS*ARwNh!4n2VrSE{9p6z+iANjPgKS4`gU5eHnZWoxUyz zwmVtSXRLrJcbN;&sH9CUOM>HWA^2Ycz~5vrUs@!nqF!|P_@G59b?K^9h8v%+f~8sp zd)tQ?z8HnhL(5-$Qi8uOiwV=!nQ$1CRM!yj)eM^CMUuy;*-ec<$6&e|1bcfjCd%K1 zp8Q0Acbp|3AA5(Y{bg$*sAFMkS5k&C z5baM!u|(0|9mg{PLA-X~Zq8tXok}^zuO@)MwI4!TI&2-ZT(^_KHKbsyG>&TS3(-!r zcl#|%Aa$o0;x>*t6T=D`(^+>%CUc1MEiRELMZGbDb&KCcJGXtvzm@1QiHs0PSVIUU zyhsQmY$R-Q%P{x*P`6#6+fVv*{WbgpjBNs7&2qh<0000bbVXQnWMOn=I%9HWVRU5x zGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK! pF)c7QEipG#F)}(bH##vkD=;uRFfbDqc((um002ovPDHLkV1kxQUw;4q literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/stock_smiley-18.png b/Media/Themes/Umami/Icon/emotes/stock_smiley-18.png new file mode 100644 index 0000000000000000000000000000000000000000..50627bff5483841c0d05583ee3bfa958f7cd3af6 GIT binary patch literal 1496 zcmV;}1t004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 KMNUMnLSTX;ij{Ey literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/stock_smiley-22.png b/Media/Themes/Umami/Icon/emotes/stock_smiley-22.png new file mode 100644 index 0000000000000000000000000000000000000000..0c87d782bf2974a320e0834eb8512b322aecb4fe GIT binary patch literal 1411 zcmV-}1$_F6P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 P00000NkvXXu0mjfYTSwy literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/stock_smiley-4.png b/Media/Themes/Umami/Icon/emotes/stock_smiley-4.png new file mode 100644 index 0000000000000000000000000000000000000000..e5f4d021da677344eafde3f61817215aa1314d2b GIT binary patch literal 1468 zcmV;t1w;CYP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 KMNUMnLSTX;ij{Ey literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/emotes/stock_smiley-8.png b/Media/Themes/Umami/Icon/emotes/stock_smiley-8.png new file mode 100644 index 0000000000000000000000000000000000000000..eb58a7beaebf64b836b3639f5579bb321c9ae13a GIT binary patch literal 1438 zcmV;P1!4M$P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00etUL_t(Y$K{qwY?VbA zhM)Q8KezUr(>sS!3TaX)V3a}x3>Ym0jTj3X8&*hMun|p+qKSqDOJyr6YLpm=m#T?T z@J2ulh6E@W!o^nFa%m|oZ9VPjx%{`8adS)#*aB|cILVjE_hsgNGT&t04E%3~F^JC2 z4%xZ0BWw)fZui}B;YZsaS|@F*-In?hja+DikhTf>m9l;fj5)OVjia9lF#j5`_l35~ zNZ5TpQB|{k@v{5MD(f3Wc})rctCrgq z-?g?bO*F4u8&=st2lumq-%$G`_an*^n6e}!D%kPnMRvT`i`Tao+a2W5wQXg} z=r4Efyjk<6v!jh7MhXzTjg$P^X z^z3Kef+b>ZL&H6~cgdC+0gl#fP4k+<{?%jXsq0uqPNdwu8|w3>vL85u=%18>KStK`sp;39d&| zV;rLtXa(93r2D9DUX2%+6*B^SWgbY>HRAVu0@^^JFj}%@V-xjr%F$!j5h01t5w>k! zOd*}Y_zp&C^khGY<@LDAPQ9|BC9&)4o~eHTTxE1_VuT7m0He^JkD8j`wWq3)5eq#! zim)w&kv!cNB}nH9(muuuFahYoB#s}5j2qF@oTG zN3o(3VM#n+WBCE5;9(pe?FA5;#poYh=)fP^_hWZ%T7c_|pZW&}*UwFb!bvBfKtRD0 z-SyXlul^(zwxiYQSE2}6mKl@g|RL0xx z#_aav!`3`gBd8Gy--m3V$otn&8lquAWz?b!L)Z^Emo9MRa@=u{o4CbUr!zaTX-(7p zQOS2T;augzb>p$QF-27fY+Ha3U<_Ir{`3Uj4<-H6;|a(0*>d2-;5UFOg@|d)PXf_d zvm&Y0&9$#bL-NV_@u|?9m?s-zE>US1E`%A%hfVinxijT*;Y?rVJnLuu$@nE&**@E4xPE0i?tWHkT) z03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R) zMObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;spF*YkOFgh?W6Bc;40000PbVXQn sQ*UN;cVTj60C#tHE@^ISb7Ns}WiD@WXPfRk8UO$Q07*qoM6N<$g3Hu>PXGV_ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/application-certificate.png b/Media/Themes/Umami/Icon/mimetypes/application-certificate.png new file mode 100644 index 0000000000000000000000000000000000000000..f09c9ec2ac0b94c791fc02215de5aff79611dda8 GIT binary patch literal 1440 zcmV;R1z-A!P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/application-vnd.ms-word.document.macroEnabled.12.png b/Media/Themes/Umami/Icon/mimetypes/application-vnd.ms-word.document.macroEnabled.12.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 zcmV+-1Ks?IP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JaQL_t(Y$JLa}YEw}Z zhQGDadTDB*peZ3kai%y@+cYtvw$SJB9V$X~phjXG_$a=Blc3UzI1=$PFw#bv7%@eu z5tE$lVDFsdm=k+qKybmi?6VH*|JVBez2Sck>0cCz`P;zFaS9q^?%cS(b}KYlDCW($ zQ7Gn3f6a+7pmMoPEEdBUgCHP+xL(D7_dg((IjK}?pjIe=En*9-Z#)|qsstwsKxXk` z=u{2_NH1hix9;|?Dy{>^UI>AV2=MUn=SbqNt|UhTP*pmf2f+E09hS~c@p9)2?<&U! z{(rOE^a=eQ8usz|i>*NN=%E&eCuW$9_Y5K;eBXVI_jBK_bp(J5=jKT!lXR_9+neRl zR-MhaC;ag>>t(#w51RG&7GQ`9Lq}AtR()$WKD?sQrwgV-`&$)&*pV5CgQ|`OAR^Wt zKxO-5&->0XUZ+jF)x?-RrcR$3szE3~7sR-^PM*pRlh%H&4H-DoBd8u zYfS+a^&J;kgD$z!6W4(R54Y}Z>B_Rr{gucfvf!R)^Pn$>pb8~YC;YN-CfaiI4??1S=Hcr8v?aH;l1pWX#4IFdVG!Vz0T`|A_ zEuaZBfUm&5TX$Uk-^4E$HpTN``*fKA001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzS zHdHYbir;gjt9pDxeJzd_w{i+uUv)zBCol zvXYY2UzkKeVf1bO~^Vjp#aybc@ zeZl(x0K;a*rvO0Uu2G%hN*8s5UukR%v2LN9+OHcxU8QZl7Bl7yE03MgPCsUcQS0N!GXfzrCuvjb( zhvRa&csw2e2m}HEgolR{i9`S(kw~vXm0G#C0F-AGGPzVHmkLFK{DR!9ED=AQ$4}>} z)XD-yUS58#OfLOcd|Dt%cRC%SEJ2O#yu~tdvJCO`DM+-__)n#wT+M4CMIk)o84}ovfHPorlzN-XJ%$*XJ;J_ zhtuht`!Hc{Zf<^leqmu@abaO`adByB$(6))xm?Sc8OzJdD=RCjtE+2kYi^;??RKxP zuWxK@yvWRaq0#(TRJ19VZz~krD%Fl!y|c69dUE{-0EpA1aCqSHny0t74+`b$yC3c6 z@Bel{05&iv_}#FGLriueFD*SYDG!U*rUpM; z45~Ko!ryVK&kuGRdaOO$jH-A=+QqucpG=Azwpc7K6GSl#?fCa&$R_-qww9Lm_IKEX zz&9AgC}LyS!$vyhEN;|fVjakXi0_v4-d#alZD%vZGXVdieLBCjn#nl**YGfB!jFHk zkuK!l=hP=X=8S0@-wYk@CuA4OS|cJ4x8>wYiI2=^tRuO@$n*EVdg&`}xAF3nWU}H9 z#Mn`!cI!~SE5?#e$`P{73;uksAV|xPj@})&hMA#}Ni=d=cClQdkpl&xASC1{35iW5 z$I*_&(8#e-2tq@UCU+cp51>@Z@^i|7f{@81B$h;`rjjW%3YA8tz5+~=lHDT%o5_j) InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/application-vnd.openxmlformats-officedocument.presentationml.template.png b/Media/Themes/Umami/Icon/mimetypes/application-vnd.openxmlformats-officedocument.presentationml.template.png new file mode 100644 index 0000000000000000000000000000000000000000..fe81b20af68ae28a9177a11f42988732b3dbd7bc GIT binary patch literal 1296 zcmV+r1@HQaP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/deb.png b/Media/Themes/Umami/Icon/mimetypes/deb.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/document.png b/Media/Themes/Umami/Icon/mimetypes/document.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/font-x-generic.png b/Media/Themes/Umami/Icon/mimetypes/font-x-generic.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/font.png b/Media/Themes/Umami/Icon/mimetypes/font.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/font_bitmap.png b/Media/Themes/Umami/Icon/mimetypes/font_bitmap.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/font_truetype.png b/Media/Themes/Umami/Icon/mimetypes/font_truetype.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/font_type1.png b/Media/Themes/Umami/Icon/mimetypes/font_type1.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-fs-executable.png b/Media/Themes/Umami/Icon/mimetypes/gnome-fs-executable.png new file mode 100644 index 0000000000000000000000000000000000000000..9843f26fe3ba8d9bfcf2540ced3fee6e8a11c9ba GIT binary patch literal 1198 zcmV;f1X25mP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Xc|L_t(Y$JLd~ZyQw< z#=m>#j%V!nS*pa(xN!?jNrIxZr36u>5~OJ*1dEDQceH^8ZFw!=PhbNQTXqm3m1V`N z6%s-sZIqBy%G0gu)K2}36MN#=nTcOBEHZ9Qh*KjZB#v}-?_ACNzVDp-7~uaNq}3N+ zJol*!KvI8nPu=`1)yjADpTMt-p51uk^^-Dyq(1lW1?uqk-X4~^++O?8@jlb}k%5*) zXe;n5mxiT)&*uonSFsdJGbL6i>-*zr@^mY9+tYiK1V2-0izT=<8bw@Gnfsp zn_nJ0`cx6RAN<kw#=`d>DyO?HiN%EIwJW! zUPm;O1EUm-QgGZ(f9|DY5O@ZGXE0ms=4+$pm1ZJzFZc)Vj7Xhsk0Z9e@z`~}r-b0U zn==^f58&8g7fO{Xyv`2eYeRj7y+r74_=oR~NIsXxk<8_Hpt|$A`wqJq4yy^8s^LVB zAI@iNxc<{Lq@rf&@9E6%7HP~NdGXDGyw~Y*tZo!`TvPg_r#o5vG9HBMpbhN@EYPYd zf}s=y#vt-M%+`Z8f4f1!lLHCXAmIM?#@%Fse;>ogdrLjC?TMPU^N-hRx)KHwCbrkl&cRC;qjv9$SYqh-iv*b&E@2P-^+(0 z8Tg!bti-aYR#nX3Pr|?$tQI3Q6faf^y9EVIgs?CMg2+_)2Q@1`}taJZ%7uDCI z_c^CLcxu43ys{1gfKmdZXu#l!04O1tU5ukp+y*5Csl-EVY9T?0R_>a-^+j~wZK_{l z?uAg=G&MObcLzEiZ%L>c?uF7|ltQhl5P617D$*3*^q|#6W?4fpuJdZX} zfY0Rs<2aa2A~MN{HZd1(3jfO*?2bfuTyC?u#w11fxF%9DoKuZB@Vm;6OIDWGE^58w%&SRNGqH?(I3sG!=Lh31jU084_Li z@j~puZ3%!8fS~hgjY4|~v;nXJumf<&g^akFd-PjLIDDZ}NR=yPg&X_fF78iX|0tIZ z6?8CbD7Ok=dp`lDJ2L6bQdaVUQp|=F-aizKtYK;5>n{sZd{Zw=4L}({Ne3(Xc<*hh zhoEOrZ#KPbZ2)-Pr3|0|U=zSbLn}6dTKOl}vk?Ip^ipU5D!Nk{z?MGu-^ZUKg^FM@ zi`EGM001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHYbir;gjt9pDxeJzd_w{i+uUv)zBCol zvXYY2UzkKeVf1bO~^Vjp#aybc@ zeZl(x0K;a*rvO0Uu2G%hN*8s5UukR%v2LN9+OHcxU8QZl7Bl7yE03MgPCsUcQS0N!GXfzrCuvjb( zhvRa&csw2e2m}HEgolR{i9`S(kw~vXm0G#C0F-AGGPzVHmkLFK{DR!9ED=AQ$4}>} z)XD-yUS58#OfLOcd|Dt%cRC%SEJ2O#yu~tdvJCO`DM+-__)n#wT+M4CMIk)o84}ovfHPorlzN-XJ%$*XJ;J_ zhtuht`!Hc{Zf<^leqmu@abaO`adByB$(6))xm?Sc8OzJdD=RCjtE+2kYi^;??RKxP zuWxK@yvWRaq0#(TRJ19VZz~krD%Fl!y|c69dUE{-0EpA1aCqSHny0t74+`b$yC3c6 z@Bel{05&iv_}#FGLriueFD*SYDG!U*rUpM; z45~Ko!ryVK&kuGRdaOO$jH-A=+QqucpG=Azwpc7K6GSl#?fCa&$R_-qww9Lm_IKEX zz&9AgC}LyS!$vyhEN;|fVjakXi0_v4-d#alZD%vZGXVdieLBCjn#nl**YGfB!jFHk zkuK!l=hP=X=8S0@-wYk@CuA4OS|cJ4x8>wYiI2=^tRuO@$n*EVdg&`}xAF3nWU}H9 z#Mn`!cI!~SE5?#e$`P{73;uksAV|xPj@})&hMA#}Ni=d=cClQdkpl&xASC1{35iW5 z$I*_&(8#e-2tq@UCU+cp51>@Z@^i|7f{@81B$h;`rjjW%3YA8tz5+~=lHDT%o5_j) InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-msword.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-msword.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 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 InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.graphics-template.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.graphics-template.png new file mode 100644 index 0000000000000000000000000000000000000000..3b7177d9b69ae0e0d14d43be24900b4e3f0b5287 GIT binary patch literal 1247 zcmV<51R(o~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00ZGkL_t(YiQSb;Y*keh zhQEKEQtoXjrcG+CSlWQbss&>V6_AHBoc^Z~TB$m`tR+k3WybI#?qwCc>nm8|TYefD1e`q%5g|3c~?da$Q2 z)$2BH+TiA{N00So)~;)u{PCw#gR381_4B!NzxJ+Jx#F9#vC+>eDk?tk{af!IX)Sm2 z=n((`^rd>;6OC&Z!oLYz1V(`2f;F$*Z53AGjPrk z1VMNRw6wGUD4X+1CX!t9r<6=2mB~%Z2{ku2OBOL`+=|PA3xMEXF);S++lQG{T$EHg z?Q3bW0)INq;-Une8TBde1Ha zbar+MVC4Hl%w&9$Rf&Ze#L~DjNApXISy7dxytIh=>Lm5mN&Rr}ZR)F&5lq}{A>Vt~ zJxB8HVwe-6Sq#j=^d^h5EG}9~d&g^}`YZGKdH%(h6=ZG-Or39aVrF?;W5>S4ONL=( zp(={y^3`$<9vvsuUl~=p%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vlD=;uRFfg@Ya{~YX002ov JPDHLkV1k#hBh~-_ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.graphics.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.graphics.png new file mode 100644 index 0000000000000000000000000000000000000000..884d386cdf1c053a8cdd2b13f4d20dc4e9ba41a1 GIT binary patch literal 883 zcmV-(1C0EMP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00MPML_t(Y$JLZiPZLoT z#eersDIGu(OqzyNf>BI}(X9)j8`maAO^6E@CVl~T1Usb(w55gkE!?4D>&hiWH?CZm zs3lMn35ya4nZnR{E@nDZX~6;;Pv#|a-+Q^|p8My){|;$2N(+TqAbTE%#+bS3DQBkb zVQHaY&WqAQ!L<5}wIixjDp;00`dMZQewpg+MMhLvAJ) zvhfzruM@Wt zOpHvDFXR!Vqk@hpQdJN^RZvk>mDTDyl2`i}yO}0A+Q({j2}>+gwL1c148HG2hmPX_ z;5A(Agw2jy!%o;7H1;t9nM?-JE^DBw7-LWoip3%#5Rci^>n?-#W!!z2-dF-lmHGL3 zI-xCrWm%!*prs84P$otuSy^Ag^S6mx3GR+Q4GS#%HM6HHSbYC0vOl~j191P@6!1LS z|F#CZE3W`3m&^ZNgC_1=Z;*N1%e9opP=Ac^VVm(`o6pOy86UR8Xb6X`3kA)dw&NbX zu|aZ$mwf!1jMP1R^aTH44^Yaba!{{sYS6^y_7KaTws7~8Q5(8j6@0IbyY&Nf49imJ z82>2*wY4wQTBq7HINS|tVL7Tng6!&`&U!7D4lPfD6=s9bhlC8_;nqE*`%DNu=`a^3u>K z0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vlD=;uRFfg@Ya{~YX002ov JPDHLkV1j9%Z(0BV literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.image.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.image.png new file mode 100644 index 0000000000000000000000000000000000000000..884d386cdf1c053a8cdd2b13f4d20dc4e9ba41a1 GIT binary patch literal 883 zcmV-(1C0EMP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00MPML_t(Y$JLZiPZLoT z#eersDIGu(OqzyNf>BI}(X9)j8`maAO^6E@CVl~T1Usb(w55gkE!?4D>&hiWH?CZm zs3lMn35ya4nZnR{E@nDZX~6;;Pv#|a-+Q^|p8My){|;$2N(+TqAbTE%#+bS3DQBkb zVQHaY&WqAQ!L<5}wIixjDp;00`dMZQewpg+MMhLvAJ) zvhfzruM@Wt zOpHvDFXR!Vqk@hpQdJN^RZvk>mDTDyl2`i}yO}0A+Q({j2}>+gwL1c148HG2hmPX_ z;5A(Agw2jy!%o;7H1;t9nM?-JE^DBw7-LWoip3%#5Rci^>n?-#W!!z2-dF-lmHGL3 zI-xCrWm%!*prs84P$otuSy^Ag^S6mx3GR+Q4GS#%HM6HHSbYC0vOl~j191P@6!1LS z|F#CZE3W`3m&^ZNgC_1=Z;*N1%e9opP=Ac^VVm(`o6pOy86UR8Xb6X`3kA)dw&NbX zu|aZ$mwf!1jMP1R^aTH44^Yaba!{{sYS6^y_7KaTws7~8Q5(8j6@0IbyY&Nf49imJ z82>2*wY4wQTBq7HINS|tVL7Tng6!&`&U!7D4lPfD6=s9bhlC8_;nqE*`%DNu=`a^3u>K z0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vlD=;uRFfg@Ya{~YX002ov JPDHLkV1j9%Z(0BV literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.presentation-template.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.presentation-template.png new file mode 100644 index 0000000000000000000000000000000000000000..fe81b20af68ae28a9177a11f42988732b3dbd7bc GIT binary patch literal 1296 zcmV+r1@HQaP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00a_AL_t(Y$Hi4`Y*S?v zKK-P%9Tld~O1cim#@2PhIyYB|+l+Ar0v*R-3KabyY}ARY@Rd^EQ_dw}QpZizJ<**MvaEd9d!xa-riiENvd;zI;Kb>e;S`+ZW5U@rT!_hG)H zbJfy4Ki0o-vz`&C)#{VG+Fxvw02k80geBmNyrTrbw&V515o*7FaIh{F!N(tc(thM< zSKGGdUVum>0t&gDX%~-10eK!*4sDsso z9hnkkXr@%c*qJkM*We%&7Ziy4fzRi|U`-90TUt<~*Q3Wff`FNkm>Zsko7a0-jEI<> zKm?z-dKIdJL0Dn202dBJEF8wwN)_f^9DrZ^eyr*0!r~1jC+N2jF;tk&||zQyw=iM8>IeZa?~pk1rU9l`MMwOj=0 zRSYX^_n}-aXXE!cw!)m;$2^@|GZ?_f+X{7{$m%apr7LxZ*;uM{H^R4bSiCP3% zYEf#=L1wIa5+<)5V>5j0ugr42T-*Qh-e(HTme#I9rWI)2l^A8$@SXXskKg+I3sDQ`Si!Av0S!3F7hiL$7rn zw1q-J{QknXOs1-;Y9x}!mae_=^M}bomq18xvyuc%vc3EEJnG!t?sT^A-g@{*cWXEt zew(CeroQ*`=`#UPrH?FOGA{haBFL05pIS|Q|J9q7Pq^_>vaG|Ae_Uvb_ zEd=U(@6>76d_og6Z`J8^r6nb$qf=8;zmJXiea{;|=QQ~iQ0cUwx!VBZU*eA|Lo5f% zba1ht#^z!x$;(Y-K;W+s_!l&O$Gox}eS2EBZ=3aaelv>uB9SNqn49*K0o)2K{xY*g zAbu0A4IPVvY-+pF#rGn zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij0000bir;gjt9pDxeJzd_w{i+uUv)zBCol zvXYY2UzkKeVf1bO~^Vjp#aybc@ zeZl(x0K;a*rvO0Uu2G%hN*8s5UukR%v2LN9+OHcxU8QZl7Bl7yE03MgPCsUcQS0N!GXfzrCuvjb( zhvRa&csw2e2m}HEgolR{i9`S(kw~vXm0G#C0F-AGGPzVHmkLFK{DR!9ED=AQ$4}>} z)XD-yUS58#OfLOcd|Dt%cRC%SEJ2O#yu~tdvJCO`DM+-__)n#wT+M4CMIk)o84}ovfHPorlzN-XJ%$*XJ;J_ zhtuht`!Hc{Zf<^leqmu@abaO`adByB$(6))xm?Sc8OzJdD=RCjtE+2kYi^;??RKxP zuWxK@yvWRaq0#(TRJ19VZz~krD%Fl!y|c69dUE{-0EpA1aCqSHny0t74+`b$yC3c6 z@Bel{05&iv_}#FGLriueFD*SYDG!U*rUpM; z45~Ko!ryVK&kuGRdaOO$jH-A=+QqucpG=Azwpc7K6GSl#?fCa&$R_-qww9Lm_IKEX zz&9AgC}LyS!$vyhEN;|fVjakXi0_v4-d#alZD%vZGXVdieLBCjn#nl**YGfB!jFHk zkuK!l=hP=X=8S0@-wYk@CuA4OS|cJ4x8>wYiI2=^tRuO@$n*EVdg&`}xAF3nWU}H9 z#Mn`!cI!~SE5?#e$`P{73;uksAV|xPj@})&hMA#}Ni=d=cClQdkpl&xASC1{35iW5 z$I*_&(8#e-2tq@UCU+cp51>@Z@^i|7f{@81B$h;`rjjW%3YA8tz5+~=lHDT%o5_j) InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.spreadsheet-template.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.oasis.opendocument.spreadsheet-template.png new file mode 100644 index 0000000000000000000000000000000000000000..711b5b37f6c89787e390ed62914c2f8960cceee7 GIT binary patch literal 1382 zcmV-s1)2JZP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00d}BL_t(Y$K6#=OdM4l z{(iGiwy-nXLSbQ-rZiYAk+w1AuRz(v#Di_s#6+Vf+w|bUgNNcx8{?6a>BS4Fu?G)I zVz8|w8WW%b%a*p*Zp(kt#m;V7w@`LwVTRp#=Dpcn#s<8a_>#A8=KcA7-}n1|Zx;Ap zw4;qLj!lHdS@)?f#>cMi&djZ;@Oo=oOZ?s+zb~FRaXd0JI~{IlI5HfKKKjWY@DC)D z$*UfZ=PT1R`-VT=&vxj;JT~zk9<;5s-HB-eSaI}(V|?Ox7jq+@mFrFCKL^-$fcsZ2 zq34TLD&jA_y}cM29>vnq5?n497_)a0$CrrsNdoa##z)SI*{cMy_eOvH0Sq&c>&j4G z_fY{z0QB|s*#XHu&f_mE%p))HaV&;_-_IkFs;Vpy@Dl@3OjQx=35tCOgKuGV@iJhc zK(GLiSC9uFVo+5z$5nvvz(hohNFr5@vD{uplD$VA8-%-x33BFo{NEMu9tZ^B@p#1i z+}s=s27`Q_B9RE1nwt1Di$o$M1kaA z)-|1Jng+hkN4_RvlWjt5d=SiHc6X#maw-TITAbepyC zK+{FuIb$)VQW^(i$W_|o-v*IZH|!KS6#S!#93aF!@W?VG5OGnm?1k;A1h3-FrH2=- zt-#;$1`-b!ft^y6S1=J8FDw{_Noh?J9`MMTmcsnd5aJg<#!&qsx^bX_qQM^AsH+wG zW8rJUWJ>_BofUMnzk;f&D&d3My@x9hrARTN`=4AS?fNx3^!^zt?s|z?5+P$*Ft>P4 zhERR-G!l_71^bNgY`v`t#=3_J=OZkloUD zuOdMo49AW&fmaCsTF;hOz*Sibq&LV&CTJ_msH~wB_4O%K$5e5~_wws_a(70Mx3A}<(jSLOAc#O^DBc!6!NKA#Way!JQM)LnBh4cg9o67*~1F$!zkk4hgs&kEvzNWP` z-~RRWRPjdoFUGKvDJ@+&FD}b})cywk<~6|`za}aG001R)MObuXVRU6WV{&C-bY%cC zFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00W#!L_t(Y$F-GDXdHDE z$3O44yWPoVXA%?4zfG%2HCnnks40y#m=#Zg;GrN^y##vDiziWps$hu5fCmqrMWIq? zkya`e+CwQRT8kj!Mom&uBAMNGt;S?F$;|G~&%?}aW3max(g%-uzc;^m-}ilAe!qdc zY*G!zC&u0b-oBG!TIc zkXx4l(WFw{bsOIVAR>qeTEn$#v$Z)cU zM5G2%U5hAeJ9m4plK%rBq*5u8$s_wtZ+nXOMp!b4jve! zv$GT1w((q#6$!c)k)Z3a+bRRLCs2Px`FtLLxDh95nFx@}<*+OhLVW@zphOXo9WYog zvgdge9R~>3PO%#EpEZcqOI864!ysvzD1>|-OtXogo}$TWs*O(Vgu!=LzU7HSLjbt0 zhZ3jOrQL_`+WGpG$w zL?{#rhzPFhA>tsSD4se=^y(xJyz)Hd2+#5ZSsp1X31wVs|IUz?tO3AhRTvGE8B5CIy2 zX21jzp~nDYWZ%Bw^y13IsU=LizfgZ_#0TQcgCx^ zD8X$T3*$)WvCtdCrw-i(ilMIn6`;)BwtoP=f|FXGaCI>N001R)MObuXVRU6WV{&C- zbY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00XE=L_t(Y$K8}oXk29! zhM)6IGUf+cqITl66iQ25#ndV_|Gyv=)K*-&RKcAa7vn+*BJK<*+O0vsg-bQ+N)aY) zIx}e!!KAclN}}ncnPhDKs!f_??(5>-nang3w}OZHnD2hvd*1VY?|TmXFXD+OPhL)@ z66);gP%@QJdq>;G=-9|oW8<0h#Kiby!>Ol!ow_+SboAKKpXTP$-$m*pU$3pLejW~o zKaE;EHilx}iz&S_G^8nOzEa#m)U8C_^ zOS+MMY*+H@^Xr8~^N9h%kNRATugDcMxH3ANKNO(q@D>kM>{DHB;R~4=_vWw3+x(jn z22JNt4&(*u6a&O!F#wtyTZl%ZU?zZtl{$ShoFz4RAaIdsJv$?BYeVbV>5?nDo%$Y0 z$Ww`3cWIExWC*P3E&qO1%Ryp5D985Qi~vGa`=GXnLE|~HlfDL<>v#UuMM6G<^z7V@ z_x+!|+_tv2)+Mc|rr}fpU~XuD?ag(n>#O!=kY^$1Teb3d=PvhzfXY%=0dTPO6|&c+ zq1+Ya86;B)+S@uXOz*xJCAfFY@R?zGmL4GI93Q_Cxai6cgsMTljmk43Qt$1kQ;tDFWHY)u zdt~PPs-4Mjm_yE-oO3ETE{OD{FR_r{#ksA&%6Bq5_%d@iXENvH%y4FM<^nK;;aSnJ ztAF+-|3+J~6zoN4CFjWWkV?}Q}fd2meWLI}5so^A@ zT^$vpVUn;Q>}PyBZB^fOOCH#K#m=fsKQ=hAcO*P;ibZ>_8>zf)6N(<|vCpbuc< z&L7;Jy-s#C#o|~BV9+;PX;c8UKn+k8^ppd(fNkLZ11raW9sdBnx%|uD6Y#|V001R) zMObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHYP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00ZhtL_t(Y$F-HqPgGYN z$3N%J+_{uTL57)O7zImc+7zk-0R?K4nl75Can){Yo5pn4f1s&Ro3zzf+HDu^T$pxY z`dHMMm~>$+3L@d5BLrkno&y)=0rR{w_nt0*ItZgQ#&2?y^E>C>-{-#9EU1>kT_E za(q7Dwt;T}RTY5lp((nDrqE+)vZjS<6Wk8=bD+`3F}HhXfWH1abR6tN z2*KdsAOPWbjvIj$YIl41;N2!(^EoNjG$Qd752tnd0@L&au3-8dp zprxe+r?;NJ0&$KVZsyY?eyS>p2#$|1`e=@Y1e6uQ$4C6USzk?X)y;`hm-julM4?a! zAp~P%V_fMDas0h~yj@>Ocqv3xZ6oDXUXtl7q3IxY%Rn(xyx&%jX-Tf%8o3B`6wDyM z$&bS}s+<)ZZfihQEOPczPz0*ojzhImtl9B6%ka8gxGG9$Yurs;O=bIuQ8~3?>v!8NJ@>W9)z()m3)YE&LcRCNP zl`5R1u6h@D#uur7-Hj!m46k@J%mIHRLI|Xk%+4>6PNnd=$^lvnX3){ziQDZ)2tnG& zp=l13Qa90f8hzCunK4;S8W?eX(-`r%oD7dnps99%%4arN*VjI_*%V@lG$rXdO6?kp zi7J*<2RYLs9nRs;X31m>%1TQ}CX?*-xmZjY0O5jwd{veeF*&nDb3--td%QfJp5^Al z6$~7tj1qqQbqGzEd~tjqtE;OFJeYv8I)J-7YI04vje*e_3~Mcpx;;LQHoKT#46__d z5X%T2M=W$*Cl-rQw9Ch6Fa&VrnSc#_{^22(BXMrsn<6-?lTN4U8yX=T&!R^XEJu?p zucU~@Vl2cmL=uvjP`LN!^PgN|{XT3dk;miNWS>0R!cV{TVp-7ghKH*|F(QToBWIE^ za%iexDk^!jn5B1MjOo$4XMkcL2W0*OSeEtO_dlFJy05UK#FfA}SEig7zF*rIiH99diD=;uR zFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgH!CnOIxsM` SVRHij0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JaQL_t(Y$JLa}YEw}Z zhQGDadTDB*peZ3kai%y@+cYtvw$SJB9V$X~phjXG_$a=Blc3UzI1=$PFw#bv7%@eu z5tE$lVDFsdm=k+qKybmi?6VH*|JVBez2Sck>0cCz`P;zFaS9q^?%cS(b}KYlDCW($ zQ7Gn3f6a+7pmMoPEEdBUgCHP+xL(D7_dg((IjK}?pjIe=En*9-Z#)|qsstwsKxXk` z=u{2_NH1hix9;|?Dy{>^UI>AV2=MUn=SbqNt|UhTP*pmf2f+E09hS~c@p9)2?<&U! z{(rOE^a=eQ8usz|i>*NN=%E&eCuW$9_Y5K;eBXVI_jBK_bp(J5=jKT!lXR_9+neRl zR-MhaC;ag>>t(#w51RG&7GQ`9Lq}AtR()$WKD?sQrwgV-`&$)&*pV5CgQ|`OAR^Wt zKxO-5&->0XUZ+jF)x?-RrcR$3szE3~7sR-^PM*pRlh%H&4H-DoBd8u zYfS+a^&J;kgD$z!6W4(R54Y}Z>B_Rr{gucfvf!R)^Pn$>pb8~YC;YN-CfaiI4??1S=Hcr8v?aH;l1pWX#4IFdVG!Vz0T`|A_ zEuaZBfUm&5TX$Uk-^4E$HpTN``*fKA001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzS zHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00eAFL_t(Y$L&;GOjCCl zSLV!o*d<{;>|uM^9%hy;PLlcLh?gwDbWowRmfq;t6&0KrXj^)r1zS)+q#$-}X)6ON zR}m|eIY~4sauaBw2sVH^1}lRMI@DJ-ni?N9h(vle0|Vt zHIDZ5bpnCgdmH)&dIQ8Cld68EI65`8x3@z79XBm-?Pd#=u2s&-6i4Tk>GFUkLq+B0 zAE&xK9pL)eNm=ct(Mzrii&zmwc*w7Ms(Qw;^n?ayJq-BKdxRYB!V0K%1!!C{(iX+>^(9`@C+7@H2Iq&sPnAA6oC;O-mN5k;=}| z!6`!?VE?gIZE*F*mA?ZQ8w;VtYDQax!SgWljA_?p=R?9rovSGK!fJzE@cSpgXgW=m z*vueTN&z#2{DM6AvD^;JeW$CknJ|C}9F^seouellIxyj*&MeBkL|VRT08_~sups%U zECmXSPC@aRA}Bvs3e8t8(XA;hvq5#80~(tfpaS1x=5X8H4jyke9RMqMh;py4s6wSq z^OF)gI$ObF`vEkWDkv&0fO1DE)SRybv&D#lWl&Xp4r*$v=qi@yL@;vC2ZW!dkoKV5 zTZk7Qt0F0R{}dV04lC920J(0&*~?s20$ZsCq-jSW4Mo3CJq+I@eg#=4ba1QVI{1bMX@A0# zE2Ps~LO#@?+@=))L}Gqd!WWWZO@Ufp268di%P-*;G#t^R&@Gw`^dr zAI5%tK>HE@bge3|dGiMqBC%j3iWQlJlItG}5FQa0DHQQXMDc=7bfz+OTEJBSckkXo zkuiFJk6QAcd?yJVNJ^wQ+}K4&rGu)gt6ezsUGjuTEHpiNQ(xn7*-8vBk9{0Hbok(e zTBD?7*c!D)L2d`Rg0$6WRUnlefyDja0-GHJELIf63HCusY6|G}dN7;Kus53Jdolni z>GjaiZ87l@;XS^9w}^A`i{gcUBz!6PT_EDm@%Y?7*_`NkR&>-{Bx}!H)ZWi#IQ!TE zp(t)C{L`>$wOS22oemt13J41i^Q_oK%RvzD?A*CM3FqRDmxyP@5+Nmy7lJ?(2SPEQ zAb^$5pm6`kq1#e-?Ao5XCvtaonnFIM(Q2mbcH4Z!?g;nO*3vq}D~LA`ZzJAEY(ad) zU@Ia7@g8#>-z4otY+!R@6mo@Z$Y3zIvC0Skv{{3gYF=i@!HA8IAei}EH!yxd%fwhN zn~$hOggkS7GsGaKs`ZRr$E<<1EB4KQ_WT6|`5kYlK0d$z001R)MObuXVRU6WV{&C- zbY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00eAFL_t(Y$L&;GOjCCl zSLV!o*d<{;>|uM^9%hy;PLlcLh?gwDbWowRmfq;t6&0KrXj^)r1zS)+q#$-}X)6ON zR}m|eIY~4sauaBw2sVH^1}lRMI@DJ-ni?N9h(vle0|Vt zHIDZ5bpnCgdmH)&dIQ8Cld68EI65`8x3@z79XBm-?Pd#=u2s&-6i4Tk>GFUkLq+B0 zAE&xK9pL)eNm=ct(Mzrii&zmwc*w7Ms(Qw;^n?ayJq-BKdxRYB!V0K%1!!C{(iX+>^(9`@C+7@H2Iq&sPnAA6oC;O-mN5k;=}| z!6`!?VE?gIZE*F*mA?ZQ8w;VtYDQax!SgWljA_?p=R?9rovSGK!fJzE@cSpgXgW=m z*vueTN&z#2{DM6AvD^;JeW$CknJ|C}9F^seouellIxyj*&MeBkL|VRT08_~sups%U zECmXSPC@aRA}Bvs3e8t8(XA;hvq5#80~(tfpaS1x=5X8H4jyke9RMqMh;py4s6wSq z^OF)gI$ObF`vEkWDkv&0fO1DE)SRybv&D#lWl&Xp4r*$v=qi@yL@;vC2ZW!dkoKV5 zTZk7Qt0F0R{}dV04lC920J(0&*~?s20$ZsCq-jSW4Mo3CJq+I@eg#=4ba1QVI{1bMX@A0# zE2Ps~LO#@?+@=))L}Gqd!WWWZO@Ufp268di%P-*;G#t^R&@Gw`^dr zAI5%tK>HE@bge3|dGiMqBC%j3iWQlJlItG}5FQa0DHQQXMDc=7bfz+OTEJBSckkXo zkuiFJk6QAcd?yJVNJ^wQ+}K4&rGu)gt6ezsUGjuTEHpiNQ(xn7*-8vBk9{0Hbok(e zTBD?7*c!D)L2d`Rg0$6WRUnlefyDja0-GHJELIf63HCusY6|G}dN7;Kus53Jdolni z>GjaiZ87l@;XS^9w}^A`i{gcUBz!6PT_EDm@%Y?7*_`NkR&>-{Bx}!H)ZWi#IQ!TE zp(t)C{L`>$wOS22oemt13J41i^Q_oK%RvzD?A*CM3FqRDmxyP@5+Nmy7lJ?(2SPEQ zAb^$5pm6`kq1#e-?Ao5XCvtaonnFIM(Q2mbcH4Z!?g;nO*3vq}D~LA`ZzJAEY(ad) zU@Ia7@g8#>-z4otY+!R@6mo@Z$Y3zIvC0Skv{{3gYF=i@!HA8IAei}EH!yxd%fwhN zn~$hOggkS7GsGaKs`ZRr$E<<1EB4KQ_WT6|`5kYlK0d$z001R)MObuXVRU6WV{&C- zbY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;wH)0002_L%V+f00eAFL_t(Y$L&;GOjCCl zSLV!o*d<{;>|uM^9%hy;PLlcLh?gwDbWowRmfq;t6&0KrXj^)r1zS)+q#$-}X)6ON zR}m|eIY~4sauaBw2sVH^1}lRMI@DJ-ni?N9h(vle0|Vt zHIDZ5bpnCgdmH)&dIQ8Cld68EI65`8x3@z79XBm-?Pd#=u2s&-6i4Tk>GFUkLq+B0 zAE&xK9pL)eNm=ct(Mzrii&zmwc*w7Ms(Qw;^n?ayJq-BKdxRYB!V0K%1!!C{(iX+>^(9`@C+7@H2Iq&sPnAA6oC;O-mN5k;=}| z!6`!?VE?gIZE*F*mA?ZQ8w;VtYDQax!SgWljA_?p=R?9rovSGK!fJzE@cSpgXgW=m z*vueTN&z#2{DM6AvD^;JeW$CknJ|C}9F^seouellIxyj*&MeBkL|VRT08_~sups%U zECmXSPC@aRA}Bvs3e8t8(XA;hvq5#80~(tfpaS1x=5X8H4jyke9RMqMh;py4s6wSq z^OF)gI$ObF`vEkWDkv&0fO1DE)SRybv&D#lWl&Xp4r*$v=qi@yL@;vC2ZW!dkoKV5 zTZk7Qt0F0R{}dV04lC920J(0&*~?s20$ZsCq-jSW4Mo3CJq+I@eg#=4ba1QVI{1bMX@A0# zE2Ps~LO#@?+@=))L}Gqd!WWWZO@Ufp268di%P-*;G#t^R&@Gw`^dr zAI5%tK>HE@bge3|dGiMqBC%j3iWQlJlItG}5FQa0DHQQXMDc=7bfz+OTEJBSckkXo zkuiFJk6QAcd?yJVNJ^wQ+}K4&rGu)gt6ezsUGjuTEHpiNQ(xn7*-8vBk9{0Hbok(e zTBD?7*c!D)L2d`Rg0$6WRUnlefyDja0-GHJELIf63HCusY6|G}dN7;Kus53Jdolni z>GjaiZ87l@;XS^9w}^A`i{gcUBz!6PT_EDm@%Y?7*_`NkR&>-{Bx}!H)ZWi#IQ!TE zp(t)C{L`>$wOS22oemt13J41i^Q_oK%RvzD?A*CM3FqRDmxyP@5+Nmy7lJ?(2SPEQ zAb^$5pm6`kq1#e-?Ao5XCvtaonnFIM(Q2mbcH4Z!?g;nO*3vq}D~LA`ZzJAEY(ad) zU@Ia7@g8#>-z4otY+!R@6mo@Z$Y3zIvC0Skv{{3gYF=i@!HA8IAei}EH!yxd%fwhN zn~$hOggkS7GsGaKs`ZRr$E<<1EB4KQ_WT6|`5kYlK0d$z001R)MObuXVRU6WV{&C- zbY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00W#!L_t(Y$F-GDXdHDE z$3O44yWPoVXA%?4zfG%2HCnnks40y#m=#Zg;GrN^y##vDiziWps$hu5fCmqrMWIq? zkya`e+CwQRT8kj!Mom&uBAMNGt;S?F$;|G~&%?}aW3max(g%-uzc;^m-}ilAe!qdc zY*G!zC&u0b-oBG!TIc zkXx4l(WFw{bsOIVAR>qeTEn$#v$Z)cU zM5G2%U5hAeJ9m4plK%rBq*5u8$s_wtZ+nXOMp!b4jve! zv$GT1w((q#6$!c)k)Z3a+bRRLCs2Px`FtLLxDh95nFx@}<*+OhLVW@zphOXo9WYog zvgdge9R~>3PO%#EpEZcqOI864!ysvzD1>|-OtXogo}$TWs*O(Vgu!=LzU7HSLjbt0 zhZ3jOrQL_`+WGpG$w zL?{#rhzPFhA>tsSD4se=^y(xJyz)Hd2+#5ZSsp1X31wVs|IUz?tO3AhRTvGE8B5CIy2 zX21jzp~nDYWZ%Bw^y13IsU=LizfgZ_#0TQcgCx^ zD8X$T3*$)WvCtdCrw-i(ilMIn6`;)BwtoP=f|FXGaCI>N001R)MObuXVRU6WV{&C- zbY%cCFfuSLFgYzSHdHYbir;gjt9pDxeJzd_w{i+uUv)zBCol zvXYY2UzkKeVf1bO~^Vjp#aybc@ zeZl(x0K;a*rvO0Uu2G%hN*8s5UukR%v2LN9+OHcxU8QZl7Bl7yE03MgPCsUcQS0N!GXfzrCuvjb( zhvRa&csw2e2m}HEgolR{i9`S(kw~vXm0G#C0F-AGGPzVHmkLFK{DR!9ED=AQ$4}>} z)XD-yUS58#OfLOcd|Dt%cRC%SEJ2O#yu~tdvJCO`DM+-__)n#wT+M4CMIk)o84}ovfHPorlzN-XJ%$*XJ;J_ zhtuht`!Hc{Zf<^leqmu@abaO`adByB$(6))xm?Sc8OzJdD=RCjtE+2kYi^;??RKxP zuWxK@yvWRaq0#(TRJ19VZz~krD%Fl!y|c69dUE{-0EpA1aCqSHny0t74+`b$yC3c6 z@Bel{05&iv_}#FGLriueFD*SYDG!U*rUpM; z45~Ko!ryVK&kuGRdaOO$jH-A=+QqucpG=Azwpc7K6GSl#?fCa&$R_-qww9Lm_IKEX zz&9AgC}LyS!$vyhEN;|fVjakXi0_v4-d#alZD%vZGXVdieLBCjn#nl**YGfB!jFHk zkuK!l=hP=X=8S0@-wYk@CuA4OS|cJ4x8>wYiI2=^tRuO@$n*EVdg&`}xAF3nWU}H9 z#Mn`!cI!~SE5?#e$`P{73;uksAV|xPj@})&hMA#}Ni=d=cClQdkpl&xASC1{35iW5 z$I*_&(8#e-2tq@UCU+cp51>@Z@^i|7f{@81B$h;`rjjW%3YA8tz5+~=lHDT%o5_j) InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.stardivision.writer.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.stardivision.writer.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 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 JPDHLkV1j9%Z(0BV literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.draw.template.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.draw.template.png new file mode 100644 index 0000000000000000000000000000000000000000..3b7177d9b69ae0e0d14d43be24900b4e3f0b5287 GIT binary patch literal 1247 zcmV<51R(o~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00ZGkL_t(YiQSb;Y*keh zhQEKEQtoXjrcG+CSlWQbss&>V6_AHBoc^Z~TB$m`tR+k3WybI#?qwCc>nm8|TYefD1e`q%5g|3c~?da$Q2 z)$2BH+TiA{N00So)~;)u{PCw#gR381_4B!NzxJ+Jx#F9#vC+>eDk?tk{af!IX)Sm2 z=n((`^rd>;6OC&Z!oLYz1V(`2f;F$*Z53AGjPrk z1VMNRw6wGUD4X+1CX!t9r<6=2mB~%Z2{ku2OBOL`+=|PA3xMEXF);S++lQG{T$EHg z?Q3bW0)INq;-Une8TBde1Ha zbar+MVC4Hl%w&9$Rf&Ze#L~DjNApXISy7dxytIh=>Lm5mN&Rr}ZR)F&5lq}{A>Vt~ zJxB8HVwe-6Sq#j=^d^h5EG}9~d&g^}`YZGKdH%(h6=ZG-Or39aVrF?;W5>S4ONL=( zp(={y^3`$<9vvsuUl~=p%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vlD=;uRFfg@Ya{~YX002ov JPDHLkV1k#hBh~-_ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.impress.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.impress.png new file mode 100644 index 0000000000000000000000000000000000000000..fd788b8638b87183e27e6fe837f5a47c155b62ba GIT binary patch literal 1007 zcmY+DeNfT|9LB%SsaUXcxry5Fkf}M0+YeGP>bir;gjt9pDxeJzd_w{i+uUv)zBCol zvXYY2UzkKeVf1bO~^Vjp#aybc@ zeZl(x0K;a*rvO0Uu2G%hN*8s5UukR%v2LN9+OHcxU8QZl7Bl7yE03MgPCsUcQS0N!GXfzrCuvjb( zhvRa&csw2e2m}HEgolR{i9`S(kw~vXm0G#C0F-AGGPzVHmkLFK{DR!9ED=AQ$4}>} z)XD-yUS58#OfLOcd|Dt%cRC%SEJ2O#yu~tdvJCO`DM+-__)n#wT+M4CMIk)o84}ovfHPorlzN-XJ%$*XJ;J_ zhtuht`!Hc{Zf<^leqmu@abaO`adByB$(6))xm?Sc8OzJdD=RCjtE+2kYi^;??RKxP zuWxK@yvWRaq0#(TRJ19VZz~krD%Fl!y|c69dUE{-0EpA1aCqSHny0t74+`b$yC3c6 z@Bel{05&iv_}#FGLriueFD*SYDG!U*rUpM; z45~Ko!ryVK&kuGRdaOO$jH-A=+QqucpG=Azwpc7K6GSl#?fCa&$R_-qww9Lm_IKEX zz&9AgC}LyS!$vyhEN;|fVjakXi0_v4-d#alZD%vZGXVdieLBCjn#nl**YGfB!jFHk zkuK!l=hP=X=8S0@-wYk@CuA4OS|cJ4x8>wYiI2=^tRuO@$n*EVdg&`}xAF3nWU}H9 z#Mn`!cI!~SE5?#e$`P{73;uksAV|xPj@})&hMA#}Ni=d=cClQdkpl&xASC1{35iW5 z$I*_&(8#e-2tq@UCU+cp51>@Z@^i|7f{@81B$h;`rjjW%3YA8tz5+~=lHDT%o5_j) InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.impress.template.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-vnd.sun.xml.impress.template.png new file mode 100644 index 0000000000000000000000000000000000000000..fe81b20af68ae28a9177a11f42988732b3dbd7bc GIT binary patch literal 1296 zcmV+r1@HQaP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-abiword.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-abiword.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-arj.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-arj.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-bzip-compressed-tar.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-bzip-compressed-tar.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-bzip.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-bzip.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-compress.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-compress.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-compressed-tar.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-compressed-tar.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-cpio-compressed.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-cpio-compressed.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-cpio.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-cpio.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-deb.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-deb.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-dvi.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-dvi.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 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 bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-bdf.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-bdf.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-linux-psf.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-linux-psf.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-pcf.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-pcf.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-sunos-news.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-sunos-news.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-ttf.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-font-ttf.png new file mode 100644 index 0000000000000000000000000000000000000000..4fc4303ee25be3648276debb9e0b71f47d2cecc5 GIT binary patch literal 1057 zcmV++1m63JP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00RC=L_t(Y$F-H)YgAVh z#(#UR(M%FEnpiD?7NdP>Uwr6XV!`55OQm36^v|f+sQ4_1rS|eah?US%R4A>2&CDc| zf@$rGQYo5bLUWnXsB>}7+54Q;hjhj{Gn33n(*=jM;OzZ<`}@{9`@sJks#aY3>hmkM zZ6DtX&`>b+?a5C+J5eXNaPfQ?sE(gHyCtXo`kPBww$*cH{G8=$RHBq3GnEGLWDy~q zN#VK<=}Zb^45{=aTH8#fe!;daCR4YutZJh#3c%OQerW8ydKMAJl864yV~7Zec#?aI z&!|TBBLazd0udqh?s{&H7J}NzYl}ss;Q`~g+?ZWqp-=%{zN^*E5oo92Mj(tf{O^xDmD-PE)w#=(pm-EIGcjEC0O^Gwkm%7eU&5c46(B>fYvU{ zPhNDisEvY*XG5)C$XBpzhc}0}Gu$6S8$+&8ZgY?h1j)o$1Nge{t_U|~m-+DEFe1X9 z9bvR~$gNgrR@sqYYC7Wq7ZL9KwZ=#^%wTU25#hDL9*l9w<%>;rb}>cCMA8FZwH>bC zSt33xUNgNs?pjow_Iwnvx_EAXQw=~fAfzO4!$wK?j1cu`$7yxg6xfMBNR~N z@+GYLxYE|576tKGvVNlcbBntV*EoIbE&M(_cI+MqlYdfXVYS4*J^kHJQFdk;5y2I> zJUP$dw|BLGi@;ES7;PLLEft$4t^+|l7Dq&wzF%N&rA+KVw55?E!p^=BT01<*J$3LpMZGQS~3ClK~reDj>b8~hP*LC@Mc99>Z=GZ9O^k7{La#ut64@>y0i~1?r78+3EBqDZS0%rX zlHczyE1$1eEK~}kqpv@on3(A7;eU;P0M=qq$R|7DlK=n!C3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij000?uMObuGZ)S9NVRB^vcXxL#X>MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-gnumeric.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-gnumeric.png new file mode 100644 index 0000000000000000000000000000000000000000..1c7bc745d6965403174e35e917e57bbf5b41e329 GIT binary patch literal 1178 zcmV;L1ZDe)P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00W#!L_t(Y$F-GDXdHDE z$3O44yWPoVXA%?4zfG%2HCnnks40y#m=#Zg;GrN^y##vDiziWps$hu5fCmqrMWIq? zkya`e+CwQRT8kj!Mom&uBAMNGt;S?F$;|G~&%?}aW3max(g%-uzc;^m-}ilAe!qdc zY*G!zC&u0b-oBG!TIc zkXx4l(WFw{bsOIVAR>qeTEn$#v$Z)cU zM5G2%U5hAeJ9m4plK%rBq*5u8$s_wtZ+nXOMp!b4jve! zv$GT1w((q#6$!c)k)Z3a+bRRLCs2Px`FtLLxDh95nFx@}<*+OhLVW@zphOXo9WYog zvgdge9R~>3PO%#EpEZcqOI864!ysvzD1>|-OtXogo}$TWs*O(Vgu!=LzU7HSLjbt0 zhZ3jOrQL_`+WGpG$w zL?{#rhzPFhA>tsSD4se=^y(xJyz)Hd2+#5ZSsp1X31wVs|IUz?tO3AhRTvGE8B5CIy2 zX21jzp~nDYWZ%Bw^y13IsU=LizfgZ_#0TQcgCx^ zD8X$T3*$)WvCtdCrw-i(ilMIn6`;)BwtoP=f|FXGaCI>N001R)MObuXVRU6WV{&C- zbY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-gzpostscript.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-gzpostscript.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 zcmV+-1Ks?IP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JaQL_t(Y$JLa}YEw}Z zhQGDadTDB*peZ3kai%y@+cYtvw$SJB9V$X~phjXG_$a=Blc3UzI1=$PFw#bv7%@eu z5tE$lVDFsdm=k+qKybmi?6VH*|JVBez2Sck>0cCz`P;zFaS9q^?%cS(b}KYlDCW($ zQ7Gn3f6a+7pmMoPEEdBUgCHP+xL(D7_dg((IjK}?pjIe=En*9-Z#)|qsstwsKxXk` z=u{2_NH1hix9;|?Dy{>^UI>AV2=MUn=SbqNt|UhTP*pmf2f+E09hS~c@p9)2?<&U! z{(rOE^a=eQ8usz|i>*NN=%E&eCuW$9_Y5K;eBXVI_jBK_bp(J5=jKT!lXR_9+neRl zR-MhaC;ag>>t(#w51RG&7GQ`9Lq}AtR()$WKD?sQrwgV-`&$)&*pV5CgQ|`OAR^Wt zKxO-5&->0XUZ+jF)x?-RrcR$3szE3~7sR-^PM*pRlh%H&4H-DoBd8u zYfS+a^&J;kgD$z!6W4(R54Y}Z>B_Rr{gucfvf!R)^Pn$>pb8~YC;YN-CfaiI4??1S=Hcr8v?aH;l1pWX#4IFdVG!Vz0T`|A_ zEuaZBfUm&5TX$Uk-^4E$HpTN``*fKA001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzS zHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-killustrator.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-killustrator.png new file mode 100644 index 0000000000000000000000000000000000000000..35b6094c35fb17e6c523022e1c0d97433637abc1 GIT binary patch literal 1063 zcmV+?1laqDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00SvWL_t(Y$L*C*XdG1> z$3Jgo-cDvG*)*Fb7*NDokXlhhL=VymUOZ@z9u-TELXT2tBVsR}6}*Jv!6H2sTPjGr zMk87=4c1z1)!6Qq9GY(Sf9AdS^RTnSPO{r5=*=GtGjD$HeLwU2{@%>M|FKIbfQ2&) z?*jAxvi5t&jvkv2(!u29C8$MlALHmM@zAK-do)+E$wRhqSR#veY6u@28;bOj0+j|x=RXZUjY z8s*6?eqAdwUM%38WKZUt;GFR1&);K>;coQ?@$@6i%ss~Mfj_zj%7aoIf>JbFG0mn> z2tdc_PU!-uIKl5x)q%W-EIOvr`u(9 z`3tlTN!s@abb-CIb8OY`V3IE0fKRy!HtP+}oV~!{i{PB2(Q4CZw6i9Mfu$HEAi{6e z%UCB|K6?p=V!BiqZc)Gdpb#Pg)>;y4F-8dV7<;D1@|8JDr+JS?vxN?0d|H$yXVHNh z4G;mXqbwP*OYEJ+BwaeKO`d-IAmLb%iK0*SPGz`7zV9<#p59G17aJQJ6bcb9zw|s) z6A@Z#R<2yZ_r2pg0M9+OpC7JW-&Itx^P8xB!8yn3kE?(zy!pn_4|0Ggig^3@>w_wJ zF*O<>B7F7r61Q%xQMrBlqxn_lWuZ-2H!f|3&--LdBr|{>EyR0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGP zFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG# hF)}(bH##vlD=;uRFfg@Ya{~YX002ovPDHLkV1nx0)nEVs literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-kpresenter.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-kpresenter.png new file mode 100644 index 0000000000000000000000000000000000000000..fd788b8638b87183e27e6fe837f5a47c155b62ba GIT binary patch literal 1007 zcmY+DeNfT|9LB%SsaUXcxry5Fkf}M0+YeGP>bir;gjt9pDxeJzd_w{i+uUv)zBCol zvXYY2UzkKeVf1bO~^Vjp#aybc@ zeZl(x0K;a*rvO0Uu2G%hN*8s5UukR%v2LN9+OHcxU8QZl7Bl7yE03MgPCsUcQS0N!GXfzrCuvjb( zhvRa&csw2e2m}HEgolR{i9`S(kw~vXm0G#C0F-AGGPzVHmkLFK{DR!9ED=AQ$4}>} z)XD-yUS58#OfLOcd|Dt%cRC%SEJ2O#yu~tdvJCO`DM+-__)n#wT+M4CMIk)o84}ovfHPorlzN-XJ%$*XJ;J_ zhtuht`!Hc{Zf<^leqmu@abaO`adByB$(6))xm?Sc8OzJdD=RCjtE+2kYi^;??RKxP zuWxK@yvWRaq0#(TRJ19VZz~krD%Fl!y|c69dUE{-0EpA1aCqSHny0t74+`b$yC3c6 z@Bel{05&iv_}#FGLriueFD*SYDG!U*rUpM; z45~Ko!ryVK&kuGRdaOO$jH-A=+QqucpG=Azwpc7K6GSl#?fCa&$R_-qww9Lm_IKEX zz&9AgC}LyS!$vyhEN;|fVjakXi0_v4-d#alZD%vZGXVdieLBCjn#nl**YGfB!jFHk zkuK!l=hP=X=8S0@-wYk@CuA4OS|cJ4x8>wYiI2=^tRuO@$n*EVdg&`}xAF3nWU}H9 z#Mn`!cI!~SE5?#e$`P{73;uksAV|xPj@})&hMA#}Ni=d=cClQdkpl&xASC1{35iW5 z$I*_&(8#e-2tq@UCU+cp51>@Z@^i|7f{@81B$h;`rjjW%3YA8tz5+~=lHDT%o5_j) InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-kspread.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-kspread.png new file mode 100644 index 0000000000000000000000000000000000000000..1c7bc745d6965403174e35e917e57bbf5b41e329 GIT binary patch literal 1178 zcmV;L1ZDe)P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00W#!L_t(Y$F-GDXdHDE z$3O44yWPoVXA%?4zfG%2HCnnks40y#m=#Zg;GrN^y##vDiziWps$hu5fCmqrMWIq? zkya`e+CwQRT8kj!Mom&uBAMNGt;S?F$;|G~&%?}aW3max(g%-uzc;^m-}ilAe!qdc zY*G!zC&u0b-oBG!TIc zkXx4l(WFw{bsOIVAR>qeTEn$#v$Z)cU zM5G2%U5hAeJ9m4plK%rBq*5u8$s_wtZ+nXOMp!b4jve! zv$GT1w((q#6$!c)k)Z3a+bRRLCs2Px`FtLLxDh95nFx@}<*+OhLVW@zphOXo9WYog zvgdge9R~>3PO%#EpEZcqOI864!ysvzD1>|-OtXogo}$TWs*O(Vgu!=LzU7HSLjbt0 zhZ3jOrQL_`+WGpG$w zL?{#rhzPFhA>tsSD4se=^y(xJyz)Hd2+#5ZSsp1X31wVs|IUz?tO3AhRTvGE8B5CIy2 zX21jzp~nDYWZ%Bw^y13IsU=LizfgZ_#0TQcgCx^ zD8X$T3*$)WvCtdCrw-i(ilMIn6`;)BwtoP=f|FXGaCI>N001R)MObuXVRU6WV{&C- zbY%cCFfuSLFgYzSHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JaQL_t(Y$JLa}YEw}Z zhQGDadTDB*peZ3kai%y@+cYtvw$SJB9V$X~phjXG_$a=Blc3UzI1=$PFw#bv7%@eu z5tE$lVDFsdm=k+qKybmi?6VH*|JVBez2Sck>0cCz`P;zFaS9q^?%cS(b}KYlDCW($ zQ7Gn3f6a+7pmMoPEEdBUgCHP+xL(D7_dg((IjK}?pjIe=En*9-Z#)|qsstwsKxXk` z=u{2_NH1hix9;|?Dy{>^UI>AV2=MUn=SbqNt|UhTP*pmf2f+E09hS~c@p9)2?<&U! z{(rOE^a=eQ8usz|i>*NN=%E&eCuW$9_Y5K;eBXVI_jBK_bp(J5=jKT!lXR_9+neRl zR-MhaC;ag>>t(#w51RG&7GQ`9Lq}AtR()$WKD?sQrwgV-`&$)&*pV5CgQ|`OAR^Wt zKxO-5&->0XUZ+jF)x?-RrcR$3szE3~7sR-^PM*pRlh%H&4H-DoBd8u zYfS+a^&J;kgD$z!6W4(R54Y}Z>B_Rr{gucfvf!R)^Pn$>pb8~YC;YN-CfaiI4??1S=Hcr8v?aH;l1pWX#4IFdVG!Vz0T`|A_ zEuaZBfUm&5TX$Uk-^4E$HpTN``*fKA001R)MObuXVRU6WV{&C-bY%cCFfuSLFgYzS zHdHY004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lhz.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lhz.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lzma-compressed-tar.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lzma-compressed-tar.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lzma.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-lzma.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-ms-dos-executable.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-ms-dos-executable.png new file mode 100644 index 0000000000000000000000000000000000000000..9843f26fe3ba8d9bfcf2540ced3fee6e8a11c9ba GIT binary patch literal 1198 zcmV;f1X25mP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 SVRHij0000Mxh&*&8$@}nIdy*OO?%l zEjVCS2!T>lZdrk?&{77JLJOs38aI`0LTQ0;X@y>1D4JlF4BgV@h3vyV?DypS{~w;5 zkEbRhJvoGQhy(y3sVVeK0C)|bKz{(RDW)C(0HKI>mIm-sUC>{g004-*kdk>8pzJ6> zMJ2!s0I2W)@DYG@HUL!$aNtUdj`FH4_l`54=rB?Q-Rcf`mzrTNAU|?`?aA;^qqtOfx4{NpBk&%(n(NUdFr`PMp#>U3S z#|;L<#KeTrXf&BjlarGVagU~^rp#vZ^z`)1^vsN9X2xQ%{5!Lj+1c4ytJOMZvn?+# zudJ-t?RJO5;druZb@j>WjijY%HVF2KF{)1$C&Zg$>Z2-T-R630nlXd;EKFyyQ_KJPmcW3D;F)FH+Q5JA7 z>2{#x1sJO8);^h=FN;pgGllJqcr#3XchO2u$PJ?C(T*(h{ZFOn_FLZ5F|p=oxjFQm z!)6n+Xkq>1fX><^!k3KcgORjwLib0+A1g@vgntI?m&MSSk8{~i-#?RjZDjtOGuWa% znBgB^vbfJy$mVtt<<{P(^T+$|+!$^QU(8RR!q=`5b_G_ktxkdBsDcurXqQu+BKO;u`BIXk3QnVNQDA@Io=jRrT z$=K004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-rpm.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-rpm.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-scribus.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-scribus.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tar.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tar.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tarz.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tarz.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tex.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-application-x-tex.png new file mode 100644 index 0000000000000000000000000000000000000000..c95c16851b0f0d9a39401b8ac8fb2752284f6957 GIT binary patch literal 802 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-audio.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-audio.png new file mode 100644 index 0000000000000000000000000000000000000000..02ed64079b6f523eea63c2557e4b20afeefde514 GIT binary patch literal 1124 zcmV-q1e^PbP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00TiuL_t(Y$L*ERYn(+G z$3M@!^S-;;tZAC8Y1*PfP>`kw9$F9ugOY6D^FB0fQq09Q9wadRRwhV zlMgq4wMPNOsOMh4TrG@`PEjtMrc|7uSQsTu%NVf|LN2-6HQjuRc4wUrK6-C=XQy)# zz#O#|g{*`Nd*rT(3oT-FBqpPfj*uiNN+5(jX*x^@L8JNza3lo)8{12ySrKbWq;?pa zl(5MV3i!~+yDmPq5##W_k1EFl2!^YlEq41oZxM+CanSdGXcU#GP=2lYd?u2RXgD4K zz&TA_DH~aMf3}9i6pI@dL#suXthqf1JHI- zHn4MN?aVy(`aJ&bM%*CEy@2Z6_~SOUA4jiGTATDMW8<{j?JP#8hyy`w5>0jPOeIO; zyCPbRHFM^Tw=?xI_|%$QHDsU~^Z-DT*srG^D@{Cgan?LMIg|GD9ooG*wZ`{!dYi-$ z#F}U@d^$CC_UY%Iebt_woTi^QWv^Y!RyV%E`6k|b$qrHwBV%LZsabC<65`SgzI9 z^4T;a?e#kBHapx}>GpqKZso0(Us(E9U%7Uzk)OX%n%&uMRZfhQ(-<;(J;U0qmRtI1 zqg!8ZFE`fQi(f1>>ww>Xrwve=dEvyHqht1Eo0*4HH8kq(yRCZoeEG-LA`k#7l|uQ# z%j%yf9wYckOJetX%I65;mIx#jYFfckWFqZQ;F#rGnC3Hnt zbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vlD=;uRFfg@Ya{~YX02y>eSaefw qW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00SvWL_t(Y$L*C*XdG1> z$3Jgo-cDvG*)*Fb7*NDokXlhhL=VymUOZ@z9u-TELXT2tBVsR}6}*Jv!6H2sTPjGr zMk87=4c1z1)!6Qq9GY(Sf9AdS^RTnSPO{r5=*=GtGjD$HeLwU2{@%>M|FKIbfQ2&) z?*jAxvi5t&jvkv2(!u29C8$MlALHmM@zAK-do)+E$wRhqSR#veY6u@28;bOj0+j|x=RXZUjY z8s*6?eqAdwUM%38WKZUt;GFR1&);K>;coQ?@$@6i%ss~Mfj_zj%7aoIf>JbFG0mn> z2tdc_PU!-uIKl5x)q%W-EIOvr`u(9 z`3tlTN!s@abb-CIb8OY`V3IE0fKRy!HtP+}oV~!{i{PB2(Q4CZw6i9Mfu$HEAi{6e z%UCB|K6?p=V!BiqZc)Gdpb#Pg)>;y4F-8dV7<;D1@|8JDr+JS?vxN?0d|H$yXVHNh z4G;mXqbwP*OYEJ+BwaeKO`d-IAmLb%iK0*SPGz`7zV9<#p59G17aJQJ6bcb9zw|s) z6A@Z#R<2yZ_r2pg0M9+OpC7JW-&Itx^P8xB!8yn3kE?(zy!pn_4|0Ggig^3@>w_wJ zF*O<>B7F7r61Q%xQMrBlqxn_lWuZ-2H!f|3&--LdBr|{>EyR0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGP zFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG# hF)}(bH##vlD=;uRFfg@Ya{~YX002ovPDHLkV1nx0)nEVs literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-html.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-html.png new file mode 100644 index 0000000000000000000000000000000000000000..d75ba938bc84dfdf37b63fb975567c6e5a789eca GIT binary patch literal 1256 zcmVP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 SVRHij0000Mxh&*&8$@}nIdy*OO?%l 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-x-zsh.png b/Media/Themes/Umami/Icon/mimetypes/gnome-mime-text-x-zsh.png new file mode 100644 index 0000000000000000000000000000000000000000..16137b7356ad3bea68bdd938a29bd46ec6ca4ad8 GIT binary patch literal 1009 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 bV_|S*E^l&Yo9;Xs00000NkvXXu0mjfH?PlO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/gnome-package.png b/Media/Themes/Umami/Icon/mimetypes/gnome-package.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/html.png b/Media/Themes/Umami/Icon/mimetypes/html.png new file mode 100644 index 0000000000000000000000000000000000000000..d75ba938bc84dfdf37b63fb975567c6e5a789eca GIT binary patch literal 1256 zcmVP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00ZhtL_t(Y$F-HqPgGYN z$3N%J+_{uTL57)O7zImc+7zk-0R?K4nl75Can){Yo5pn4f1s&Ro3zzf+HDu^T$pxY z`dHMMm~>$+3L@d5BLrkno&y)=0rR{w_nt0*ItZgQ#&2?y^E>C>-{-#9EU1>kT_E za(q7Dwt;T}RTY5lp((nDrqE+)vZjS<6Wk8=bD+`3F}HhXfWH1abR6tN z2*KdsAOPWbjvIj$YIl41;N2!(^EoNjG$Qd752tnd0@L&au3-8dp zprxe+r?;NJ0&$KVZsyY?eyS>p2#$|1`e=@Y1e6uQ$4C6USzk?X)y;`hm-julM4?a! zAp~P%V_fMDas0h~yj@>Ocqv3xZ6oDXUXtl7q3IxY%Rn(xyx&%jX-Tf%8o3B`6wDyM z$&bS}s+<)ZZfihQEOPczPz0*ojzhImtl9B6%ka8gxGG9$Yurs;O=bIuQ8~3?>v!8NJ@>W9)z()m3)YE&LcRCNP zl`5R1u6h@D#uur7-Hj!m46k@J%mIHRLI|Xk%+4>6PNnd=$^lvnX3){ziQDZ)2tnG& zp=l13Qa90f8hzCunK4;S8W?eX(-`r%oD7dnps99%%4arN*VjI_*%V@lG$rXdO6?kp zi7J*<2RYLs9nRs;X31m>%1TQ}CX?*-xmZjY0O5jwd{veeF*&nDb3--td%QfJp5^Al z6$~7tj1qqQbqGzEd~tjqtE;OFJeYv8I)J-7YI04vje*e_3~Mcpx;;LQHoKT#46__d z5X%T2M=W$*Cl-rQw9Ch6Fa&VrnSc#_{^22(BXMrsn<6-?lTN4U8yX=T&!R^XEJu?p zucU~@Vl2cmL=uvjP`LN!^PgN|{XT3dk;miNWS>0R!cV{TVp-7ghKH*|F(QToBWIE^ za%iexDk^!jn5B1MjOo$4XMkcL2W0*OSeEtO_dlFJy05UK#FfA}SEig7zF*rIiH99diD=;uR zFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgH!CnOIxsM` SVRHij0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00SvWL_t(Y$L*C*XdG1> z$3Jgo-cDvG*)*Fb7*NDokXlhhL=VymUOZ@z9u-TELXT2tBVsR}6}*Jv!6H2sTPjGr zMk87=4c1z1)!6Qq9GY(Sf9AdS^RTnSPO{r5=*=GtGjD$HeLwU2{@%>M|FKIbfQ2&) z?*jAxvi5t&jvkv2(!u29C8$MlALHmM@zAK-do)+E$wRhqSR#veY6u@28;bOj0+j|x=RXZUjY z8s*6?eqAdwUM%38WKZUt;GFR1&);K>;coQ?@$@6i%ss~Mfj_zj%7aoIf>JbFG0mn> z2tdc_PU!-uIKl5x)q%W-EIOvr`u(9 z`3tlTN!s@abb-CIb8OY`V3IE0fKRy!HtP+}oV~!{i{PB2(Q4CZw6i9Mfu$HEAi{6e z%UCB|K6?p=V!BiqZc)Gdpb#Pg)>;y4F-8dV7<;D1@|8JDr+JS?vxN?0d|H$yXVHNh z4G;mXqbwP*OYEJ+BwaeKO`d-IAmLb%iK0*SPGz`7zV9<#p59G17aJQJ6bcb9zw|s) z6A@Z#R<2yZ_r2pg0M9+OpC7JW-&Itx^P8xB!8yn3kE?(zy!pn_4|0Ggig^3@>w_wJ zF*O<>B7F7r61Q%xQMrBlqxn_lWuZ-2H!f|3&--LdBr|{>EyR0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGP zFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG# hF)}(bH##vlD=;uRFfg@Ya{~YX002ovPDHLkV1nx0)nEVs literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/image.png b/Media/Themes/Umami/Icon/mimetypes/image.png new file mode 100644 index 0000000000000000000000000000000000000000..35b6094c35fb17e6c523022e1c0d97433637abc1 GIT binary patch literal 1063 zcmV+?1laqDP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00SvWL_t(Y$L*C*XdG1> z$3Jgo-cDvG*)*Fb7*NDokXlhhL=VymUOZ@z9u-TELXT2tBVsR}6}*Jv!6H2sTPjGr zMk87=4c1z1)!6Qq9GY(Sf9AdS^RTnSPO{r5=*=GtGjD$HeLwU2{@%>M|FKIbfQ2&) z?*jAxvi5t&jvkv2(!u29C8$MlALHmM@zAK-do)+E$wRhqSR#veY6u@28;bOj0+j|x=RXZUjY z8s*6?eqAdwUM%38WKZUt;GFR1&);K>;coQ?@$@6i%ss~Mfj_zj%7aoIf>JbFG0mn> z2tdc_PU!-uIKl5x)q%W-EIOvr`u(9 z`3tlTN!s@abb-CIb8OY`V3IE0fKRy!HtP+}oV~!{i{PB2(Q4CZw6i9Mfu$HEAi{6e z%UCB|K6?p=V!BiqZc)Gdpb#Pg)>;y4F-8dV7<;D1@|8JDr+JS?vxN?0d|H$yXVHNh z4G;mXqbwP*OYEJ+BwaeKO`d-IAmLb%iK0*SPGz`7zV9<#p59G17aJQJ6bcb9zw|s) z6A@Z#R<2yZ_r2pg0M9+OpC7JW-&Itx^P8xB!8yn3kE?(zy!pn_4|0Ggig^3@>w_wJ zF*O<>B7F7r61Q%xQMrBlqxn_lWuZ-2H!f|3&--LdBr|{>EyR0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGP zFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG# hF)}(bH##vlD=;uRFfg@Ya{~YX002ovPDHLkV1nx0)nEVs literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/kpresenter_kpr.png b/Media/Themes/Umami/Icon/mimetypes/kpresenter_kpr.png new file mode 100644 index 0000000000000000000000000000000000000000..fd788b8638b87183e27e6fe837f5a47c155b62ba GIT binary patch literal 1007 zcmY+DeNfT|9LB%SsaUXcxry5Fkf}M0+YeGP>bir;gjt9pDxeJzd_w{i+uUv)zBCol zvXYY2UzkKeVf1bO~^Vjp#aybc@ zeZl(x0K;a*rvO0Uu2G%hN*8s5UukR%v2LN9+OHcxU8QZl7Bl7yE03MgPCsUcQS0N!GXfzrCuvjb( zhvRa&csw2e2m}HEgolR{i9`S(kw~vXm0G#C0F-AGGPzVHmkLFK{DR!9ED=AQ$4}>} z)XD-yUS58#OfLOcd|Dt%cRC%SEJ2O#yu~tdvJCO`DM+-__)n#wT+M4CMIk)o84}ovfHPorlzN-XJ%$*XJ;J_ zhtuht`!Hc{Zf<^leqmu@abaO`adByB$(6))xm?Sc8OzJdD=RCjtE+2kYi^;??RKxP zuWxK@yvWRaq0#(TRJ19VZz~krD%Fl!y|c69dUE{-0EpA1aCqSHny0t74+`b$yC3c6 z@Bel{05&iv_}#FGLriueFD*SYDG!U*rUpM; z45~Ko!ryVK&kuGRdaOO$jH-A=+QqucpG=Azwpc7K6GSl#?fCa&$R_-qww9Lm_IKEX zz&9AgC}LyS!$vyhEN;|fVjakXi0_v4-d#alZD%vZGXVdieLBCjn#nl**YGfB!jFHk zkuK!l=hP=X=8S0@-wYk@CuA4OS|cJ4x8>wYiI2=^tRuO@$n*EVdg&`}xAF3nWU}H9 z#Mn`!cI!~SE5?#e$`P{73;uksAV|xPj@})&hMA#}Ni=d=cClQdkpl&xASC1{35iW5 z$I*_&(8#e-2tq@UCU+cp51>@Z@^i|7f{@81B$h;`rjjW%3YA8tz5+~=lHDT%o5_j) InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/mime_ascii.png b/Media/Themes/Umami/Icon/mimetypes/mime_ascii.png new file mode 100644 index 0000000000000000000000000000000000000000..37dcf9ed9fe6a41addbcf5515eebdbc90e6bf649 GIT binary patch literal 651 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/package.png b/Media/Themes/Umami/Icon/mimetypes/package.png new file mode 100644 index 0000000000000000000000000000000000000000..2dd3e654fbe1fa32df9b31566262ff0487d2318a GIT binary patch literal 847 zcmV-V1F-ywP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY4c7nw4c7reD4Tcy00L1-L_t(Y$L*B8Yg17e zho5uqM@-s;#){SIqK@K62SIS^Abw0via7XDEP{)XqJMx=um;_PRyybsK?D~q6rs>X zutS<6E)5hFO4{)5e+lKxuiYGBZ!X;zE&Pv54!=Q^-5y^A0~|a(w+X%Jk%Wj6ElCzkTFV&r^W3 zUofWc@%?gN8X+wv-ah2;$s4pEI*Zl{X*aEFv^2w2V^vy2FF9x{!t6|rGhIoB27A(g zSOt*FjsbA6^E%3z0Ikqk;c11|9%xmGRo<0X{@IToIChTA=&LGt1u#6&4M11#h`+2G zQ|4X&dO#wjMC#VABhxdp;~tilJ32&g2mIM;I`9lgmm}VC5rB9sVqEE;SGBh39pJ$x zCIE0Zky3rDgIWhopT2OLpV^UyTx0t?8GiYQL`qe8nzfScP=uyfYeN9b4%Lj;0fZs8 zy&-@MZTl-g05Je7+=SU6pk!5M&E)XsrQ()`p^dGgt)HHV9Y& zT+GP8v-{_}8!F%L(!!E{4mf}pY)dU!Kp2Pu5g_CjWBA_{Py!YK7bpV1{8*Lh0ct6j zyP3njrE2{KZvfU3bR6Ri0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;m zIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vl ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/package_editors.png b/Media/Themes/Umami/Icon/mimetypes/package_editors.png new file mode 100644 index 0000000000000000000000000000000000000000..37dcf9ed9fe6a41addbcf5515eebdbc90e6bf649 GIT binary patch literal 651 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/shellscript.png b/Media/Themes/Umami/Icon/mimetypes/shellscript.png new file mode 100644 index 0000000000000000000000000000000000000000..16137b7356ad3bea68bdd938a29bd46ec6ca4ad8 GIT binary patch literal 1009 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/stock_calendar.png b/Media/Themes/Umami/Icon/mimetypes/stock_calendar.png new file mode 100644 index 0000000000000000000000000000000000000000..33ea6a3ee697cf20eed140c5eb871994cb33def5 GIT binary patch literal 1212 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/template_source.png b/Media/Themes/Umami/Icon/mimetypes/template_source.png new file mode 100644 index 0000000000000000000000000000000000000000..5b4b141ce72a5c260e5d6d57d795c8a0ee51e72b GIT binary patch literal 747 zcmbV~SxD0X0Ed6e>2#(gNZQCNW>_)!Z{^XtoJ`y-#bZ(?#pc=Nkxoz(vZ;9l6$&XB z8aN~_#VHvuOTTh&WlQ;0`TJVxG4bq z_jGpwpz0VI0zl5=ByfNaPdyzXR{+6Typ#l>;uug>1MC2((gCFepcew{R=}_5MQ`#s z03JoW^b){JwI?YitgQ^dRMKTZEdH1Lf;)8$HT8|PkLs)I8*2v!z70wTq|!mDY)~eb z%H=ZokbHP}NHH=zGO8FI8y!=Qj*lzHCzRiROsG_oGcz-4wR(1Tc5ZHNetuq~(P*{W zg@uL1#l@wiC7n*Uyu7T}>kS6O%F2q-Xk1-gHJMCnYisN4>t?fgV`IZ&u~@Cv&CSiN zt*!0tZJW)uv$JEj+Z_&v)9L(tUC{!-Pn?pN3er9ova7oX)%&3DA=;6k;E(U5;$xEi-IUhdJPJSE=lFCWjdR+WY*CUcmft*2 z5k7o5eGNc=CtweyS%I<|w=sTZd5h?+2-GHe_(d;qJEw^r)J*E5_|nt5dLDEj=~;ZS zRa`=@lF;YO7EieHKi+(7+KLB_TU_4#>PEnCQW%dlSz$|Ub8nBX>{dMabzR=eBrQ-_8x9DE~PkKQes5-;c-GRabNy4<==7!Y-3Qdtql*$64 zTPb*-5yCsmON%vWdelZXdH)FyP1L40fNjb&`Mg>a7~}&WR2+L%oGlXPu(OJDMB*H< z2o}L0&M=4w0W*qyDxA%XI7tv}f_T0W#r!A8FU-D`TMib%WHN{d1~XE?WU*P1Y-Z%& Spi5M;H!=9JiQIQFLdhSaJ~DRz literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/text-html.png b/Media/Themes/Umami/Icon/mimetypes/text-html.png new file mode 100644 index 0000000000000000000000000000000000000000..d75ba938bc84dfdf37b63fb975567c6e5a789eca GIT binary patch literal 1256 zcmVP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00ZhtL_t(Y$F-HqPgGYN z$3N%J+_{uTL57)O7zImc+7zk-0R?K4nl75Can){Yo5pn4f1s&Ro3zzf+HDu^T$pxY z`dHMMm~>$+3L@d5BLrkno&y)=0rR{w_nt0*ItZgQ#&2?y^E>C>-{-#9EU1>kT_E za(q7Dwt;T}RTY5lp((nDrqE+)vZjS<6Wk8=bD+`3F}HhXfWH1abR6tN z2*KdsAOPWbjvIj$YIl41;N2!(^EoNjG$Qd752tnd0@L&au3-8dp zprxe+r?;NJ0&$KVZsyY?eyS>p2#$|1`e=@Y1e6uQ$4C6USzk?X)y;`hm-julM4?a! zAp~P%V_fMDas0h~yj@>Ocqv3xZ6oDXUXtl7q3IxY%Rn(xyx&%jX-Tf%8o3B`6wDyM z$&bS}s+<)ZZfihQEOPczPz0*ojzhImtl9B6%ka8gxGG9$Yurs;O=bIuQ8~3?>v!8NJ@>W9)z()m3)YE&LcRCNP zl`5R1u6h@D#uur7-Hj!m46k@J%mIHRLI|Xk%+4>6PNnd=$^lvnX3){ziQDZ)2tnG& zp=l13Qa90f8hzCunK4;S8W?eX(-`r%oD7dnps99%%4arN*VjI_*%V@lG$rXdO6?kp zi7J*<2RYLs9nRs;X31m>%1TQ}CX?*-xmZjY0O5jwd{veeF*&nDb3--td%QfJp5^Al z6$~7tj1qqQbqGzEd~tjqtE;OFJeYv8I)J-7YI04vje*e_3~Mcpx;;LQHoKT#46__d z5X%T2M=W$*Cl-rQw9Ch6Fa&VrnSc#_{^22(BXMrsn<6-?lTN4U8yX=T&!R^XEJu?p zucU~@Vl2cmL=uvjP`LN!^PgN|{XT3dk;miNWS>0R!cV{TVp-7ghKH*|F(QToBWIE^ za%iexDk^!jn5B1MjOo$4XMkcL2W0*OSeEtO_dlFJy05UK#FfA}SEig7zF*rIiH99diD=;uR zFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgH!CnOIxsM` SVRHij00002#(gNZQCNW>_)!Z{^XtoJ`y-#bZ(?#pc=Nkxoz(vZ;9l6$&XB z8aN~_#VHvuOTTh&WlQ;0`TJVxG4bq z_jGpwpz0VI0zl5=ByfNaPdyzXR{+6Typ#l>;uug>1MC2((gCFepcew{R=}_5MQ`#s z03JoW^b){JwI?YitgQ^dRMKTZEdH1Lf;)8$HT8|PkLs)I8*2v!z70wTq|!mDY)~eb z%H=ZokbHP}NHH=zGO8FI8y!=Qj*lzHCzRiROsG_oGcz-4wR(1Tc5ZHNetuq~(P*{W zg@uL1#l@wiC7n*Uyu7T}>kS6O%F2q-Xk1-gHJMCnYisN4>t?fgV`IZ&u~@Cv&CSiN zt*!0tZJW)uv$JEj+Z_&v)9L(tUC{!-Pn?pN3er9ova7oX)%&3DA=;6k;E(U5;$xEi-IUhdJPJSE=lFCWjdR+WY*CUcmft*2 z5k7o5eGNc=CtweyS%I<|w=sTZd5h?+2-GHe_(d;qJEw^r)J*E5_|nt5dLDEj=~;ZS zRa`=@lF;YO7EieHKi+(7+KLB_TU_4#>PEnCQW%dlSz$|Ub8nBX>{dMabzR=eBrQ-_8x9DE~PkKQes5-;c-GRabNy4<==7!Y-3Qdtql*$64 zTPb*-5yCsmON%vWdelZXdH)FyP1L40fNjb&`Mg>a7~}&WR2+L%oGlXPu(OJDMB*H< z2o}L0&M=4w0W*qyDxA%XI7tv}f_T0W#r!A8FU-D`TMib%WHN{d1~XE?WU*P1Y-Z%& Spi5M;H!=9JiQIQFLdhSaJ~DRz literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/text-x-generic.png b/Media/Themes/Umami/Icon/mimetypes/text-x-generic.png new file mode 100644 index 0000000000000000000000000000000000000000..37dcf9ed9fe6a41addbcf5515eebdbc90e6bf649 GIT binary patch literal 651 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/txt.png b/Media/Themes/Umami/Icon/mimetypes/txt.png new file mode 100644 index 0000000000000000000000000000000000000000..37dcf9ed9fe6a41addbcf5515eebdbc90e6bf649 GIT binary patch literal 651 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/video-x-generic.png b/Media/Themes/Umami/Icon/mimetypes/video-x-generic.png new file mode 100644 index 0000000000000000000000000000000000000000..c9dbcb2d0a658550e5eca6e0e00704de81e9af9c GIT binary patch literal 1386 zcmV-w1(o`VP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/x-office-calendar.png b/Media/Themes/Umami/Icon/mimetypes/x-office-calendar.png new file mode 100644 index 0000000000000000000000000000000000000000..33ea6a3ee697cf20eed140c5eb871994cb33def5 GIT binary patch literal 1212 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 JPDHLkV1k#hBh~-_ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/x-office-drawing.png b/Media/Themes/Umami/Icon/mimetypes/x-office-drawing.png new file mode 100644 index 0000000000000000000000000000000000000000..884d386cdf1c053a8cdd2b13f4d20dc4e9ba41a1 GIT binary patch literal 883 zcmV-(1C0EMP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00MPML_t(Y$JLZiPZLoT z#eersDIGu(OqzyNf>BI}(X9)j8`maAO^6E@CVl~T1Usb(w55gkE!?4D>&hiWH?CZm zs3lMn35ya4nZnR{E@nDZX~6;;Pv#|a-+Q^|p8My){|;$2N(+TqAbTE%#+bS3DQBkb zVQHaY&WqAQ!L<5}wIixjDp;00`dMZQewpg+MMhLvAJ) zvhfzruM@Wt zOpHvDFXR!Vqk@hpQdJN^RZvk>mDTDyl2`i}yO}0A+Q({j2}>+gwL1c148HG2hmPX_ z;5A(Agw2jy!%o;7H1;t9nM?-JE^DBw7-LWoip3%#5Rci^>n?-#W!!z2-dF-lmHGL3 zI-xCrWm%!*prs84P$otuSy^Ag^S6mx3GR+Q4GS#%HM6HHSbYC0vOl~j191P@6!1LS z|F#CZE3W`3m&^ZNgC_1=Z;*N1%e9opP=Ac^VVm(`o6pOy86UR8Xb6X`3kA)dw&NbX zu|aZ$mwf!1jMP1R^aTH44^Yaba!{{sYS6^y_7KaTws7~8Q5(8j6@0IbyY&Nf49imJ z82>2*wY4wQTBq7HINS|tVL7Tng6!&`&U!7D4lPfD6=s9bhlC8_;nqE*`%DNu=`a^3u>K z0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vlD=;uRFfg@Ya{~YX002ov JPDHLkV1j9%Z(0BV literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/x-office-presentation-template.png b/Media/Themes/Umami/Icon/mimetypes/x-office-presentation-template.png new file mode 100644 index 0000000000000000000000000000000000000000..fe81b20af68ae28a9177a11f42988732b3dbd7bc GIT binary patch literal 1296 zcmV+r1@HQaP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00a_AL_t(Y$Hi4`Y*S?v zKK-P%9Tld~O1cim#@2PhIyYB|+l+Ar0v*R-3KabyY}ARY@Rd^EQ_dw}QpZizJ<**MvaEd9d!xa-riiENvd;zI;Kb>e;S`+ZW5U@rT!_hG)H zbJfy4Ki0o-vz`&C)#{VG+Fxvw02k80geBmNyrTrbw&V515o*7FaIh{F!N(tc(thM< zSKGGdUVum>0t&gDX%~-10eK!*4sDsso z9hnkkXr@%c*qJkM*We%&7Ziy4fzRi|U`-90TUt<~*Q3Wff`FNkm>Zsko7a0-jEI<> zKm?z-dKIdJL0Dn202dBJEF8wwN)_f^9DrZ^eyr*0!r~1jC+N2jF;tk&||zQyw=iM8>IeZa?~pk1rU9l`MMwOj=0 zRSYX^_n}-aXXE!cw!)m;$2^@|GZ?_f+X{7{$m%apr7LxZ*;uM{H^R4bSiCP3% zYEf#=L1wIa5+<)5V>5j0ugr42T-*Qh-e(HTme#I9rWI)2l^A8$@SXXskKg+I3sDQ`Si!Av0S!3F7hiL$7rn zw1q-J{QknXOs1-;Y9x}!mae_=^M}bomq18xvyuc%vc3EEJnG!t?sT^A-g@{*cWXEt zew(CeroQ*`=`#UPrH?FOGA{haBFL05pIS|Q|J9q7Pq^_>vaG|Ae_Uvb_ zEd=U(@6>76d_og6Z`J8^r6nb$qf=8;zmJXiea{;|=QQ~iQ0cUwx!VBZU*eA|Lo5f% zba1ht#^z!x$;(Y-K;W+s_!l&O$Gox}eS2EBZ=3aaelv>uB9SNqn49*K0o)2K{xY*g zAbu0A4IPVvY-+pF#rGn zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgH!CnOIxsM`VRHij0000bir;gjt9pDxeJzd_w{i+uUv)zBCol zvXYY2UzkKeVf1bO~^Vjp#aybc@ zeZl(x0K;a*rvO0Uu2G%hN*8s5UukR%v2LN9+OHcxU8QZl7Bl7yE03MgPCsUcQS0N!GXfzrCuvjb( zhvRa&csw2e2m}HEgolR{i9`S(kw~vXm0G#C0F-AGGPzVHmkLFK{DR!9ED=AQ$4}>} z)XD-yUS58#OfLOcd|Dt%cRC%SEJ2O#yu~tdvJCO`DM+-__)n#wT+M4CMIk)o84}ovfHPorlzN-XJ%$*XJ;J_ zhtuht`!Hc{Zf<^leqmu@abaO`adByB$(6))xm?Sc8OzJdD=RCjtE+2kYi^;??RKxP zuWxK@yvWRaq0#(TRJ19VZz~krD%Fl!y|c69dUE{-0EpA1aCqSHny0t74+`b$yC3c6 z@Bel{05&iv_}#FGLriueFD*SYDG!U*rUpM; z45~Ko!ryVK&kuGRdaOO$jH-A=+QqucpG=Azwpc7K6GSl#?fCa&$R_-qww9Lm_IKEX zz&9AgC}LyS!$vyhEN;|fVjakXi0_v4-d#alZD%vZGXVdieLBCjn#nl**YGfB!jFHk zkuK!l=hP=X=8S0@-wYk@CuA4OS|cJ4x8>wYiI2=^tRuO@$n*EVdg&`}xAF3nWU}H9 z#Mn`!cI!~SE5?#e$`P{73;uksAV|xPj@})&hMA#}Ni=d=cClQdkpl&xASC1{35iW5 z$I*_&(8#e-2tq@UCU+cp51>@Z@^i|7f{@81B$h;`rjjW%3YA8tz5+~=lHDT%o5_j) InIW$DANfP0g#Z8m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/mimetypes/x-office-spreadsheet-template.png b/Media/Themes/Umami/Icon/mimetypes/x-office-spreadsheet-template.png new file mode 100644 index 0000000000000000000000000000000000000000..711b5b37f6c89787e390ed62914c2f8960cceee7 GIT binary patch literal 1382 zcmV-s1)2JZP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 ZD=;uRFfg@Ya{~YX002ovPDHLkV1fduR(AjZ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/application-x-gnome-saved-search.png b/Media/Themes/Umami/Icon/places/application-x-gnome-saved-search.png new file mode 100644 index 0000000000000000000000000000000000000000..4b39b7e00835e83bef8edadd8e2589fa2e26eaaa GIT binary patch literal 1207 zcmV;o1W5adP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WdsL_t(Y$JLcPY+OYY zhQFD+4?Am{b%X)igai-}L;*-aauXgA4HBXOHy}$%OX=j^o(AyX$@2nK_5z?%Lk9jY34jsb=n+`+w(k=feNpOlvVZIvNcR z5BDDCSS%JxQ&Us&$biYo$={7J&jWgpqlm1i>Ko_JpWl2Wz{JFaj*X2GLO@j!NrP2Y zj4{m4&erDV=ckIr;`CmD#l=McE?l@!|3|?1__&OXjhU5|6}vPY9t6psFNE!pO)-cYl9Bg+hT`E{8D&Yb_$e z%*@PN2W7AW@ZPESE){nTMD|0fNsVfG3uDs6t+hLlkkrgKpT5#FQ2ouneS*YHA#H#p zaX9D6uU%*O>gO06wMEh@M{`HVAup&JQ`Rjo%q`gaN*FSrdou7ELLeJ2@#N_bcb7_;a zNlbL47bFFkq0`tGKZeROGWP|ta0%0O6npY47`rQ;lpr`lqY9RtkFldffhdWIy8BQ` z(g>bDgT^aHVxrKgbYMjIu}+ON*vUv8b>x4u*Ku3 zMreGS=-Kz6|7DUJ-y-F^*r-ghb>AVfwhsUZ^%!zT@Z}guEMW1gf8l=olI+lH*nA)E z!DZZ?YnZOb$UpZcCf`f;Hq)$8FWQhbW?9;ba!ELWdL_~cP?peYja~^aAhuU Va%Y?FJQ@H1002ovPDHLkV1o8p5Rw1@ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/desktop.png b/Media/Themes/Umami/Icon/places/desktop.png new file mode 100644 index 0000000000000000000000000000000000000000..1748a80a24f235493c8f61f3465e68b7c3027f14 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00S0DL_t(Y$L*C(XjE4e z$A2^PG-|CzjncZbDs*8tDpGA5gbHqK(HJcizp5n0g=k$UWaqj?6k@wbaV0KVsoa2h!vA9f z)C)gkJg;fnwk_m`M*w)>-(}X6?7-_n?;BKFTE5|F{}984B2W4r(tWLa8887T0LF72 zMv5i+2Z!-}~I)rL~Qo|L#*5DTUj|10w)lZB%|P zho8Mk`@Uv+`?7ITRWL>n5kw3|3}S>`J3i*{k3XYVmg8TRG;Ad~L|Jzm7 z6;nk-Pz?zL6+u)GQFd-$%#qHcjGHtwk;Zg52!is7t}B<=xBKhFCbd8kg4(%SH6p_D zrPIhi_bKEDI^Bkd2yXmwa>Mk=k7#J9uL6%ERwwds9uljN&L|D1aG|u5nbdjK-MA9oHCeD3W zXsg87ARMd0j1(+tY&!mAXf_k;-a)X^RS-3ff=a|HCa#H^NWruGd(8jnlh52XjR?}| z^t$Htn`zr~Fzyy{qjg#{N}pE7DqQ_%kguB8Q9tAT4yz(nTLa2Lz>1ZtIdtTl1=hIG zF0dst#-K(xc_z!ww!Pf;heD`vAwi-B#d4WtjbC!G<9C|Bm`f(3QPGPpUXUV1sazrm zDm`)G&&O=q+{%^fJ+-M%666a-o)6>!nDfCrjvoJ)nNxg5vwX3h&sVIa%Yyel?oaHA z2t-WJ*)yj;bUzmI!$19fvHKYC)V7}IoELAwl9jE$oj$*#D$V864V;tgUKEpnY4#n~ z#!$HrM!OR{D!=9lK*sUu#s02cf|LbLTr8uk)%rnvz>+uGD-?|d{B^!#^?NgW3HIkP zk1ltdiU0rrHFQN-bVF}#ZDnqB04QTAATls8G$2Z0Yjt8EQ*>o%Ze?-`3PW;bVRU6= zAa`kWXdqN*WgtgMO;C{8i*En`03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@ zGc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;sp bF*qwQFgh?WLn#C500000NkvXXu0mjf=dR}D literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/distributor-logo.png b/Media/Themes/Umami/Icon/places/distributor-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..9403d6a0a6fa868c02a7aece7fb9499a31c846e4 GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00TTpL_t(Y$K91ph+9<@ z$A9;|yqAy3$K-Q#zM8hm2nDS;x|)U9g^(^!r!d%S+yS*Tu9Yl$l9rb>mr{bMF6l&$;)W3;(f+#}!yE zlrIBEfQyw<{@T9^c=6rxmro?8&d$%oS^aI>^*njBQp*25n&W$noqrmg@}GU@%nJl` zAA_Nz{`AN0N5Hw!JmpaVs;a#A)~nC5W7yoN8mydL#HT8+J)HMQfaOBDkcbAfbYhBI zyIpE+n?y8-uBnmbLiy-KEz0JKQQ#XP#1Sdw8`ZBad_NLC^-MM`ZZ|DFx#uu?4s)4^ zT-$8E4tz4c0wHdOBcYd;j?c{?#MNxBxH1ye29EbY96%A!?F|7MV_THX6&HMpSbXlp zagwP7OHa)!2=QGuSG-^G^__Z0`hCg+aYYCkMo$30PI3kpQi+)C*)D5a2B}n>SU9Kv zAKr^!*{yekuI(*PMg6Sr7`Rf}l~Vr4u>c{&o5|F)FuFFLl++E2+1ZQ`LOFjguL>v~^2taVTYalc<@FmwTEn*$=@ARr^lg>neE6pIAt*)EB&&R<(i z>V|0pg-HRfWe-u6y@#$Ug8sm{AA3I*@h7u$anZ9KLb}Se>+86#_jaX}_a+52%&vuB zSNE=rPr)=TfEMul;%tJ2CzAkFw;S~OLx8LM3GEAzQhs)))|8=u27pghFfAJ?<+`GX zCBLR&m;;0mIF5^|Dt9ZT{Px5E;9}n%cx|IgB&0KRJv5(&5Msge+I=|?_@H`^i(v4ra-;cpw()d$=Ue~umWsWO8MUp(>Q!wOWA50chj-y zX$G!GJQb(eYKJPNd=L1V32fXnz2rC!j_WdXJfsvLr5G6*V{Pn@mdzD^4u(P}l9@DS zr-RXKS?gCm2u%u@ISW5%au}|nNXZ_XDy3001R)MObuX zVRU6WV{&C-bY%cCFfuSLFgYzSHdHYHq)$8FWQhbW?9;ba!EL lWdL_~cP?peYja~^aAhuUa%Y?FJQ@H1002ovPDHLkV1kUF@CpC` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/emptytrash.png b/Media/Themes/Umami/Icon/places/emptytrash.png new file mode 100644 index 0000000000000000000000000000000000000000..ae881de70b6e5e2a4a911e462eb88d14c62ca846 GIT binary patch literal 1323 zcmV+`1=RY9P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00an0L_t(Y$JLe3Yh6VY z$3OSZ{qgR5FTa}9{vgyUT67^+(S;xgR*BV?r@NWg`HxX@MqfEJ2Bu&AJJx~O2? z73)si7h1s@Z4-)V(zNEim%P6BZtjnnGvlI3nl=?`#f1lkIh@1Hch1arz6bvA=B+#9 zlhJ6kI`H}G>a`a)$N$l0o-c7+J(}me_fMWEH9)YQOp?A&o_PGFvDaTa=*^NEpToA0PQ8K79Bi+RID$zJJqD%D*M1w&b2Y zaDZxE`1G?k=Dt1F+6&+VC>2aSwtviaIt6FW{(RT>?VND$p8X6C>=+)Oj9%CPh@#}} z=bw8kId$q3BId3Elv1?Y4PJTqgL=I-ba3MhM5MaB+~C!T3C73A36#R!IM#K4eZGTO zr_)8OY})|PW%kY&U%h$!-<*E41wmwuDc@r=9jD%{HOI!D8IGeE5r+|nh(#Bcq7;f; zC<|dVGpzOuS*FP{K@1qNSaFC2Z8DPTy8+AwJZgtqD`~|YaL2esUFRVr1Js!XN=x;Q0ZSs2^i2A{wO>MUj;vc5w?pS<2-+FGU53q5$8I@co3*oloQY z6)w*GM4GP9Y|V1z>IHC$FsP!-E?Sp3K&gm4Uo}=(+5%w3HF}-2h{FDAB_6@^LOiM{ zKc?9_kB7pD!5G1arLVS~O1up#3#4fea40{Z*S#jzx~-3FEse57;%JCGZ(@wb_cUqN zWT`bxX#`_R@InUrceB)*W1wzHS62|L*96a_n|2U;y9z9gmG(6UAXXz*DD(Hq)$8FWQhbW?9;ba!ELWdL_~ hcP?peYja~^aAhuUa%Y?FJQ@H1002ovPDHLkV1hDmP>BEl literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/folder-remote.png b/Media/Themes/Umami/Icon/places/folder-remote.png new file mode 100644 index 0000000000000000000000000000000000000000..ad2f227f8ae86d5dd2870f1d51bd349f075d931e GIT binary patch literal 1128 zcmV-u1eg1XP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00TuyL_t(Y$F-KvPb5_k z$G`RZ$1pIvHo-k82RCc72ahHih%rVFoQyXTI81_xCyyLBaFHWiNsJ^OK(6bBf52=^ zNWeXqXf%q6U^46KEXy!3-822VUX_O#hUuN&bs<`(*ZtnBPkpPZ-)rE1Ho6$h&dz$1 zlan`|wJ?gJvz3(3Tn?bBu-0N~YN|XwK8~@mF@#|VB7!jnB7%*L zjrWQa^a0G7GG}nk!Dg3$K}c#DXXi19y26byeMoAVW!^7bnj1}Du< zrcUZZ005Gt4yC}uX`8`UbPfP%rf9TM3!*1Q0JJ)SS0-*E8ud{bxmt??fFVL0rwlCs z5Rv@)sEy->kNXdP%kNZywx8r)QF$DoTzqgUh~rk^K@Nui#+ZA*{Pg33n9w`MSCKmQVdKvkQ_o6pSH-rnwCFd_oy z92OQ9ngDVDRek7r-pG)G=NQe*%zP;#ALH`SCeQP}o}Qkbzs`@qe`@~#5uW^067-Jc z0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vmD=;uRFfce uSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00WdsL_t(Y$JLcPY+OYY zhQFD+4?Am{b%X)igai-}L;*-aauXgA4HBXOHy}$%OX=j^o(AyX$@2nK_5z?%Lk9jY34jsb=n+`+w(k=feNpOlvVZIvNcR z5BDDCSS%JxQ&Us&$biYo$={7J&jWgpqlm1i>Ko_JpWl2Wz{JFaj*X2GLO@j!NrP2Y zj4{m4&erDV=ckIr;`CmD#l=McE?l@!|3|?1__&OXjhU5|6}vPY9t6psFNE!pO)-cYl9Bg+hT`E{8D&Yb_$e z%*@PN2W7AW@ZPESE){nTMD|0fNsVfG3uDs6t+hLlkkrgKpT5#FQ2ouneS*YHA#H#p zaX9D6uU%*O>gO06wMEh@M{`HVAup&JQ`Rjo%q`gaN*FSrdou7ELLeJ2@#N_bcb7_;a zNlbL47bFFkq0`tGKZeROGWP|ta0%0O6npY47`rQ;lpr`lqY9RtkFldffhdWIy8BQ` z(g>bDgT^aHVxrKgbYMjIu}+ON*vUv8b>x4u*Ku3 zMreGS=-Kz6|7DUJ-y-F^*r-ghb>AVfwhsUZ^%!zT@Z}guEMW1gf8l=olI+lH*nA)E z!DZZ?YnZOb$UpZcCf`f;Hq)$8FWQhbW?9;ba!ELWdL_~cP?peYja~^aAhuU Va%Y?FJQ@H1002ovPDHLkV1o8p5Rw1@ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/folder.png b/Media/Themes/Umami/Icon/places/folder.png new file mode 100644 index 0000000000000000000000000000000000000000..5257570163e78d84b432d958481822a5e1395de3 GIT binary patch literal 1008 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00PZPL_t(Y$JLd;ZzDw% z$3L@P+ZQKyas(;3G$)Ym4?qG%LI^=kNlO0%lqrIghLouzAyTHlnic^fMJNsmL=lSQ zBm+kh;@Gj*yEAXzqp+PgwgcyghHo{inf-p|{n)qgf0r9A*4NidtE;OI?>bno*E`$W z+j?)n*4EZ9&bhY$n{`lCn`ZXT&dyHjUVzQbO=~n7#27I%R0o$eGn{iA9v=3Oj*hnL z_4@9(!O6)90Q>v<`9A_SHa4`;Xt-vxiHM-8sOmK!O;ZjI4*DU4%IF<48^6SNySuv| zO#;%O_PX<1K)h;k{>VZEazBd6~tq?+xuWv1lItOU?PDS zBR<>!#vz3lI#)fM8w8#t$u*?#;?_U?V(Bca$v014c!mve4e&k?f^d8;Jo`~dl4O{{ z(l8%4x3Y;oFjK7jp+npy6?pfLr)NK4kHOhk1Y$8pPOp^r-+sc!?>)uFe4OJ>jct`O zFMR&x*DK&`ssrB?^^a$%uT!h69bw59-q6h$%*OSSS~^A-G~8d(vM zs4P^<x`oHNzh(jG=E@tdJ)&MqS$Z&EivlFh(d!jrY8R7$)H!EIFAL7wIp01%p4%KC zn$qoM73sBJOaiJ|mVNcpuV%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 z05UK!F)c7QEipG#F)}(bH##vmD=;uRFfceSaefwW^{L9a%BK_cXuvn eZfkR6VQ^(GZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00TEkL_t(Y$JLd+Z(CIu zho5tNZ6{9BI?#aH2&vGj3k(PWiiA)_XGSvmA7IQ7q%36290`#zduLWuDl!BlVBkwZ zD5(_MVkdFz_+H<8zFr2~*RJDGHDcj69qF9&d)}{mF8tr628-q8<-+3P;@m|CtJP|I zeSJM#8nCvu_N!9r79f)jh7b-#O0m1U+uhsS zTd!8Do1+E?2L}LbZ*Qmn3RqcL3AI{H9UdOyoFjxl2q6bVQN+&9PS;vn8r%_)(I6gf zZf@Qi2SkH~_g+E>$~lMk{uzLXV2ojDX=!?Xex8|`8A_!RN-4C~gb>);+PXU_gB;+T z73VD8dyE|dMj@GTw9jIc>LsqV&LNp`YWdSIW*bSxJ-GSE&>x5#V2s6D$NrJy{*Ml= zb-#jzem#a!$=Do-1f=+p!l_eA@cxrKjqi{fpkyQiC*FJNXMqpjxyi>L+(AZsl;T2- zY!wpY`25SS7eL9m9ysSnGGSr5Oud!z*GYG>q2d|%KvTs6*XD}og6>S{ffMH(v4uBg z${-MFea_^yYZdOjwZMKoX6N@~d>E2~KxRBz>DD=bwY?@MR&k|LMg*e55KxXZ@7|v0 z)~hq16t}O>@zPwGhrcvvcO7x+m@XAC#-TLLbAVonJc)tXO0l=u_kZPTnYUk`XR4^_ zq%Mb+BE{V|7ufmjh*oTvE)_|;2H3WoTF<5DVE_OC literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/gnome-fs-desktop.png b/Media/Themes/Umami/Icon/places/gnome-fs-desktop.png new file mode 100644 index 0000000000000000000000000000000000000000..1748a80a24f235493c8f61f3465e68b7c3027f14 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00S0DL_t(Y$L*C(XjE4e z$A2^PG-|CzjncZbDs*8tDpGA5gbHqK(HJcizp5n0g=k$UWaqj?6k@wbaV0KVsoa2h!vA9f z)C)gkJg;fnwk_m`M*w)>-(}X6?7-_n?;BKFTE5|F{}984B2W4r(tWLa8887T0LF72 zMv5i+2Z!-}~I)rL~Qo|L#*5DTUj|10w)lZB%|P zho8Mk`@Uv+`?7ITRWL>n5kw3|3}S>`J3i*{k3XYVmg8TRG;Ad~L|Jzm7 z6;nk-Pz?zL6+u)GQFd-$%#qHcjGHtwk;Zg52!is7t}B<=xBKhFCbd8kg4(%SH6p_D zrPIhi_bKEDI^Bkd2yXmwa>Mk=k7#J9uL6%ERwwds9uljN&L|D1aG|u5nbdjK-MA9oHCeD3W zXsg87ARMd0j1(+tY&!mAXf_k;-a)X^RS-3ff=a|HCa#H^NWruGd(8jnlh52XjR?}| z^t$Htn`zr~Fzyy{qjg#{N}pE7DqQ_%kguB8Q9tAT4yz(nTLa2Lz>1ZtIdtTl1=hIG zF0dst#-K(xc_z!ww!Pf;heD`vAwi-B#d4WtjbC!G<9C|Bm`f(3QPGPpUXUV1sazrm zDm`)G&&O=q+{%^fJ+-M%666a-o)6>!nDfCrjvoJ)nNxg5vwX3h&sVIa%Yyel?oaHA z2t-WJ*)yj;bUzmI!$19fvHKYC)V7}IoELAwl9jE$oj$*#D$V864V;tgUKEpnY4#n~ z#!$HrM!OR{D!=9lK*sUu#s02cf|LbLTr8uk)%rnvz>+uGD-?|d{B^!#^?NgW3HIkP zk1ltdiU0rrHFQN-bVF}#ZDnqB04QTAATls8G$2Z0Yjt8EQ*>o%Ze?-`3PW;bVRU6= zAa`kWXdqN*WgtgMO;C{8i*En`03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@ zGc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;sp bF*qwQFgh?WLn#C500000NkvXXu0mjf=dR}D literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/gnome-fs-directory.png b/Media/Themes/Umami/Icon/places/gnome-fs-directory.png new file mode 100644 index 0000000000000000000000000000000000000000..5257570163e78d84b432d958481822a5e1395de3 GIT binary patch literal 1008 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/gnome-fs-network.png b/Media/Themes/Umami/Icon/places/gnome-fs-network.png new file mode 100644 index 0000000000000000000000000000000000000000..f47b6a310298d356cba268da68ffa9e9595ba16f GIT binary patch literal 1071 zcmbW0Z!nt&9LInD4Y3?^x-!=-vm-KdeV#Z$&UN)CM_NoI>2&5HCCDb2JWON~e>T0Y zm#eL(bdz4KJw2-3St7xqBph0)pqcoa>m1u5VoH0=YL~gb*xP;XzIWd@-_N`6rR1bI zBH=9p01%n+u_*xHZHtQo0L--N9|C}5Zd76vK*trg1D+EAkh0=a5&_Bs04l!**av{h zKLAP)fVE5jdLzK`qNW~JBmlS;#h(`exOv)QH?=4|0I(}aPT(-$|00t~0YE6s2LKGi z0D#SAAIWGm+EKw?L7@b`sx~+rj!Y(%N-MFVn0?d|RB>l+vt9vmFhYPC9@Zs?I| zbaZ5FY)r2o*X#8LgJE)Va%yU7dV1PuH2ymGbY^D8WHQap&d$%zFDxu9EiIYN=H=z( zm6es%)zvi{>+9?58yg!Ii)C|j(`vPDZEbCDZ$GoKvuoq|;o+fG>&pTF6i-QFgPpyD zqqD2Khu5(;k9+&R{f^&z{y{-#NO*WefpAdT7 zJ=m*Nw+@`J?w{RUY>;H!r=@XYPZrekq-Uu6dA}Vh1pxOt$B-WZWuixrPO?bcgbdj5?H-Wnw zSpMT4643G@Wr>t78}5h@alLn5%>C`Yf8h$j0p*XDB9>ruN K#omw3to{et+`YO0 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/gnome-fs-nfs.png b/Media/Themes/Umami/Icon/places/gnome-fs-nfs.png new file mode 100644 index 0000000000000000000000000000000000000000..ad2f227f8ae86d5dd2870f1d51bd349f075d931e GIT binary patch literal 1128 zcmV-u1eg1XP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/gnome-main-menu.png b/Media/Themes/Umami/Icon/places/gnome-main-menu.png new file mode 100644 index 0000000000000000000000000000000000000000..9403d6a0a6fa868c02a7aece7fb9499a31c846e4 GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00TTpL_t(Y$K91ph+9<@ z$A9;|yqAy3$K-Q#zM8hm2nDS;x|)U9g^(^!r!d%S+yS*Tu9Yl$l9rb>mr{bMF6l&$;)W3;(f+#}!yE zlrIBEfQyw<{@T9^c=6rxmro?8&d$%oS^aI>^*njBQp*25n&W$noqrmg@}GU@%nJl` zAA_Nz{`AN0N5Hw!JmpaVs;a#A)~nC5W7yoN8mydL#HT8+J)HMQfaOBDkcbAfbYhBI zyIpE+n?y8-uBnmbLiy-KEz0JKQQ#XP#1Sdw8`ZBad_NLC^-MM`ZZ|DFx#uu?4s)4^ zT-$8E4tz4c0wHdOBcYd;j?c{?#MNxBxH1ye29EbY96%A!?F|7MV_THX6&HMpSbXlp zagwP7OHa)!2=QGuSG-^G^__Z0`hCg+aYYCkMo$30PI3kpQi+)C*)D5a2B}n>SU9Kv zAKr^!*{yekuI(*PMg6Sr7`Rf}l~Vr4u>c{&o5|F)FuFFLl++E2+1ZQ`LOFjguL>v~^2taVTYalc<@FmwTEn*$=@ARr^lg>neE6pIAt*)EB&&R<(i z>V|0pg-HRfWe-u6y@#$Ug8sm{AA3I*@h7u$anZ9KLb}Se>+86#_jaX}_a+52%&vuB zSNE=rPr)=TfEMul;%tJ2CzAkFw;S~OLx8LM3GEAzQhs)))|8=u27pghFfAJ?<+`GX zCBLR&m;;0mIF5^|Dt9ZT{Px5E;9}n%cx|IgB&0KRJv5(&5Msge+I=|?_@H`^i(v4ra-;cpw()d$=Ue~umWsWO8MUp(>Q!wOWA50chj-y zX$G!GJQb(eYKJPNd=L1V32fXnz2rC!j_WdXJfsvLr5G6*V{Pn@mdzD^4u(P}l9@DS zr-RXKS?gCm2u%u@ISW5%au}|nNXZ_XDy3001R)MObuX zVRU6WV{&C-bY%cCFfuSLFgYzSHdHYHq)$8FWQhbW?9;ba!EL lWdL_~cP?peYja~^aAhuUa%Y?FJQ@H1002ovPDHLkV1kUF@CpC` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/gnome-mime-x-directory-nfs-server.png b/Media/Themes/Umami/Icon/places/gnome-mime-x-directory-nfs-server.png new file mode 100644 index 0000000000000000000000000000000000000000..cb27e4230a2aa396716f2d665f06236bf3ecd4bd GIT binary patch literal 1021 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 K#omw3to{et+`YO0 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/gnome-stock-trash.png b/Media/Themes/Umami/Icon/places/gnome-stock-trash.png new file mode 100644 index 0000000000000000000000000000000000000000..ae881de70b6e5e2a4a911e462eb88d14c62ca846 GIT binary patch literal 1323 zcmV+`1=RY9P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00an0L_t(Y$JLe3Yh6VY z$3OSZ{qgR5FTa}9{vgyUT67^+(S;xgR*BV?r@NWg`HxX@MqfEJ2Bu&AJJx~O2? z73)si7h1s@Z4-)V(zNEim%P6BZtjnnGvlI3nl=?`#f1lkIh@1Hch1arz6bvA=B+#9 zlhJ6kI`H}G>a`a)$N$l0o-c7+J(}me_fMWEH9)YQOp?A&o_PGFvDaTa=*^NEpToA0PQ8K79Bi+RID$zJJqD%D*M1w&b2Y zaDZxE`1G?k=Dt1F+6&+VC>2aSwtviaIt6FW{(RT>?VND$p8X6C>=+)Oj9%CPh@#}} z=bw8kId$q3BId3Elv1?Y4PJTqgL=I-ba3MhM5MaB+~C!T3C73A36#R!IM#K4eZGTO zr_)8OY})|PW%kY&U%h$!-<*E41wmwuDc@r=9jD%{HOI!D8IGeE5r+|nh(#Bcq7;f; zC<|dVGpzOuS*FP{K@1qNSaFC2Z8DPTy8+AwJZgtqD`~|YaL2esUFRVr1Js!XN=x;Q0ZSs2^i2A{wO>MUj;vc5w?pS<2-+FGU53q5$8I@co3*oloQY z6)w*GM4GP9Y|V1z>IHC$FsP!-E?Sp3K&gm4Uo}=(+5%w3HF}-2h{FDAB_6@^LOiM{ zKc?9_kB7pD!5G1arLVS~O1up#3#4fea40{Z*S#jzx~-3FEse57;%JCGZ(@wb_cUqN zWT`bxX#`_R@InUrceB)*W1wzHS62|L*96a_n|2U;y9z9gmG(6UAXXz*DD(Hq)$8FWQhbW?9;ba!ELWdL_~ hcP?peYja~^aAhuUa%Y?FJQ@H1002ovPDHLkV1hDmP>BEl literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/gtk-directory.png b/Media/Themes/Umami/Icon/places/gtk-directory.png new file mode 100644 index 0000000000000000000000000000000000000000..5257570163e78d84b432d958481822a5e1395de3 GIT binary patch literal 1008 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00PZPL_t(Y$JLd;ZzDw% z$3L@P+ZQKyas(;3G$)Ym4?qG%LI^=kNlO0%lqrIghLouzAyTHlnic^fMJNsmL=lSQ zBm+kh;@Gj*yEAXzqp+PgwgcyghHo{inf-p|{n)qgf0r9A*4NidtE;OI?>bno*E`$W z+j?)n*4EZ9&bhY$n{`lCn`ZXT&dyHjUVzQbO=~n7#27I%R0o$eGn{iA9v=3Oj*hnL z_4@9(!O6)90Q>v<`9A_SHa4`;Xt-vxiHM-8sOmK!O;ZjI4*DU4%IF<48^6SNySuv| zO#;%O_PX<1K)h;k{>VZEazBd6~tq?+xuWv1lItOU?PDS zBR<>!#vz3lI#)fM8w8#t$u*?#;?_U?V(Bca$v014c!mve4e&k?f^d8;Jo`~dl4O{{ z(l8%4x3Y;oFjK7jp+npy6?pfLr)NK4kHOhk1Y$8pPOp^r-+sc!?>)uFe4OJ>jct`O zFMR&x*DK&`ssrB?^^a$%uT!h69bw59-q6h$%*OSSS~^A-G~8d(vM zs4P^<x`oHNzh(jG=E@tdJ)&MqS$Z&EivlFh(d!jrY8R7$)H!EIFAL7wIp01%p4%KC zn$qoM73sBJOaiJ|mVNcpuV%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 z05UK!F)c7QEipG#F)}(bH##vmD=;uRFfceSaefwW^{L9a%BK_cXuvn eZfkR6VQ^(GZ*pgw?mQX*00002&5HCCDb2JWON~e>T0Y zm#eL(bdz4KJw2-3St7xqBph0)pqcoa>m1u5VoH0=YL~gb*xP;XzIWd@-_N`6rR1bI zBH=9p01%n+u_*xHZHtQo0L--N9|C}5Zd76vK*trg1D+EAkh0=a5&_Bs04l!**av{h zKLAP)fVE5jdLzK`qNW~JBmlS;#h(`exOv)QH?=4|0I(}aPT(-$|00t~0YE6s2LKGi z0D#SAAIWGm+EKw?L7@b`sx~+rj!Y(%N-MFVn0?d|RB>l+vt9vmFhYPC9@Zs?I| zbaZ5FY)r2o*X#8LgJE)Va%yU7dV1PuH2ymGbY^D8WHQap&d$%zFDxu9EiIYN=H=z( zm6es%)zvi{>+9?58yg!Ii)C|j(`vPDZEbCDZ$GoKvuoq|;o+fG>&pTF6i-QFgPpyD zqqD2Khu5(;k9+&R{f^&z{y{-#NO*WefpAdT7 zJ=m*Nw+@`J?w{RUY>;H!r=@XYPZrekq-Uu6dA}Vh1pxOt$B-WZWuixrPO?bcgbdj5?H-Wnw zSpMT4643G@Wr>t78}5h@alLn5%>C`Yf8h$j0p*XDB9>ruN K#omw3to{et+`YO0 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/inode-directory.png b/Media/Themes/Umami/Icon/places/inode-directory.png new file mode 100644 index 0000000000000000000000000000000000000000..5257570163e78d84b432d958481822a5e1395de3 GIT binary patch literal 1008 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 K#omw3to{et+`YO0 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/network.png b/Media/Themes/Umami/Icon/places/network.png new file mode 100644 index 0000000000000000000000000000000000000000..ad2f227f8ae86d5dd2870f1d51bd349f075d931e GIT binary patch literal 1128 zcmV-u1eg1XP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00TuyL_t(Y$F-KvPb5_k z$G`RZ$1pIvHo-k82RCc72ahHih%rVFoQyXTI81_xCyyLBaFHWiNsJ^OK(6bBf52=^ zNWeXqXf%q6U^46KEXy!3-822VUX_O#hUuN&bs<`(*ZtnBPkpPZ-)rE1Ho6$h&dz$1 zlan`|wJ?gJvz3(3Tn?bBu-0N~YN|XwK8~@mF@#|VB7!jnB7%*L zjrWQa^a0G7GG}nk!Dg3$K}c#DXXi19y26byeMoAVW!^7bnj1}Du< zrcUZZ005Gt4yC}uX`8`UbPfP%rf9TM3!*1Q0JJ)SS0-*E8ud{bxmt??fFVL0rwlCs z5Rv@)sEy->kNXdP%kNZywx8r)QF$DoTzqgUh~rk^K@Nui#+ZA*{Pg33n9w`MSCKmQVdKvkQ_o6pSH-rnwCFd_oy z92OQ9ngDVDRek7r-pG)G=NQe*%zP;#ALH`SCeQP}o}Qkbzs`@qe`@~#5uW^067-Jc z0000bbVXQnWMOn=I%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGn zC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(bH##vmD=;uRFfce uSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX*00002&5HCCDb2JWON~e>T0Y zm#eL(bdz4KJw2-3St7xqBph0)pqcoa>m1u5VoH0=YL~gb*xP;XzIWd@-_N`6rR1bI zBH=9p01%n+u_*xHZHtQo0L--N9|C}5Zd76vK*trg1D+EAkh0=a5&_Bs04l!**av{h zKLAP)fVE5jdLzK`qNW~JBmlS;#h(`exOv)QH?=4|0I(}aPT(-$|00t~0YE6s2LKGi z0D#SAAIWGm+EKw?L7@b`sx~+rj!Y(%N-MFVn0?d|RB>l+vt9vmFhYPC9@Zs?I| zbaZ5FY)r2o*X#8LgJE)Va%yU7dV1PuH2ymGbY^D8WHQap&d$%zFDxu9EiIYN=H=z( zm6es%)zvi{>+9?58yg!Ii)C|j(`vPDZEbCDZ$GoKvuoq|;o+fG>&pTF6i-QFgPpyD zqqD2Khu5(;k9+&R{f^&z{y{-#NO*WefpAdT7 zJ=m*Nw+@`J?w{RUY>;H!r=@XYPZrekq-Uu6dA}Vh1pxOt$B-WZWuixrPO?bcgbdj5?H-Wnw zSpMT4643G@Wr>t78}5h@alLn5%>C`Yf8h$j0p*XDB9>ruN K#omw3to{et+`YO0 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/novell-button.png b/Media/Themes/Umami/Icon/places/novell-button.png new file mode 100644 index 0000000000000000000000000000000000000000..9403d6a0a6fa868c02a7aece7fb9499a31c846e4 GIT binary patch literal 1119 zcmV-l1fctgP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00TTpL_t(Y$K91ph+9<@ z$A9;|yqAy3$K-Q#zM8hm2nDS;x|)U9g^(^!r!d%S+yS*Tu9Yl$l9rb>mr{bMF6l&$;)W3;(f+#}!yE zlrIBEfQyw<{@T9^c=6rxmro?8&d$%oS^aI>^*njBQp*25n&W$noqrmg@}GU@%nJl` zAA_Nz{`AN0N5Hw!JmpaVs;a#A)~nC5W7yoN8mydL#HT8+J)HMQfaOBDkcbAfbYhBI zyIpE+n?y8-uBnmbLiy-KEz0JKQQ#XP#1Sdw8`ZBad_NLC^-MM`ZZ|DFx#uu?4s)4^ zT-$8E4tz4c0wHdOBcYd;j?c{?#MNxBxH1ye29EbY96%A!?F|7MV_THX6&HMpSbXlp zagwP7OHa)!2=QGuSG-^G^__Z0`hCg+aYYCkMo$30PI3kpQi+)C*)D5a2B}n>SU9Kv zAKr^!*{yekuI(*PMg6Sr7`Rf}l~Vr4u>c{&o5|F)FuFFLl++E2+1ZQ`LOFjguL>v~^2taVTYalc<@FmwTEn*$=@ARr^lg>neE6pIAt*)EB&&R<(i z>V|0pg-HRfWe-u6y@#$Ug8sm{AA3I*@h7u$anZ9KLb}Se>+86#_jaX}_a+52%&vuB zSNE=rPr)=TfEMul;%tJ2CzAkFw;S~OLx8LM3GEAzQhs)))|8=u27pghFfAJ?<+`GX zCBLR&m;;0mIF5^|Dt9ZT{Px5E;9}n%cx|IgB&0KRJv5(&5Msge+I=|?_@H`^i(v4ra-;cpw()d$=Ue~umWsWO8MUp(>Q!wOWA50chj-y zX$G!GJQb(eYKJPNd=L1V32fXnz2rC!j_WdXJfsvLr5G6*V{Pn@mdzD^4u(P}l9@DS zr-RXKS?gCm2u%u@ISW5%au}|nNXZ_XDy3001R)MObuX zVRU6WV{&C-bY%cCFfuSLFgYzSHdHYHq)$8FWQhbW?9;ba!EL lWdL_~cP?peYja~^aAhuUa%Y?FJQ@H1002ovPDHLkV1kUF@CpC` literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/redhat-network-server.png b/Media/Themes/Umami/Icon/places/redhat-network-server.png new file mode 100644 index 0000000000000000000000000000000000000000..cb27e4230a2aa396716f2d665f06236bf3ecd4bd GIT binary patch literal 1021 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/stock_folder.png b/Media/Themes/Umami/Icon/places/stock_folder.png new file mode 100644 index 0000000000000000000000000000000000000000..5257570163e78d84b432d958481822a5e1395de3 GIT binary patch literal 1008 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00PZPL_t(Y$JLd;ZzDw% z$3L@P+ZQKyas(;3G$)Ym4?qG%LI^=kNlO0%lqrIghLouzAyTHlnic^fMJNsmL=lSQ zBm+kh;@Gj*yEAXzqp+PgwgcyghHo{inf-p|{n)qgf0r9A*4NidtE;OI?>bno*E`$W z+j?)n*4EZ9&bhY$n{`lCn`ZXT&dyHjUVzQbO=~n7#27I%R0o$eGn{iA9v=3Oj*hnL z_4@9(!O6)90Q>v<`9A_SHa4`;Xt-vxiHM-8sOmK!O;ZjI4*DU4%IF<48^6SNySuv| zO#;%O_PX<1K)h;k{>VZEazBd6~tq?+xuWv1lItOU?PDS zBR<>!#vz3lI#)fM8w8#t$u*?#;?_U?V(Bca$v014c!mve4e&k?f^d8;Jo`~dl4O{{ z(l8%4x3Y;oFjK7jp+npy6?pfLr)NK4kHOhk1Y$8pPOp^r-+sc!?>)uFe4OJ>jct`O zFMR&x*DK&`ssrB?^^a$%uT!h69bw59-q6h$%*OSSS~^A-G~8d(vM zs4P^<x`oHNzh(jG=E@tdJ)&MqS$Z&EivlFh(d!jrY8R7$)H!EIFAL7wIp01%p4%KC zn$qoM73sBJOaiJ|mVNcpuV%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 z05UK!F)c7QEipG#F)}(bH##vmD=;uRFfceSaefwW^{L9a%BK_cXuvn eZfkR6VQ^(GZ*pgw?mQX*0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00an0L_t(Y$JLe3Yh6VY z$3OSZ{qgR5FTa}9{vgyUT67^+(S;xgR*BV?r@NWg`HxX@MqfEJ2Bu&AJJx~O2? z73)si7h1s@Z4-)V(zNEim%P6BZtjnnGvlI3nl=?`#f1lkIh@1Hch1arz6bvA=B+#9 zlhJ6kI`H}G>a`a)$N$l0o-c7+J(}me_fMWEH9)YQOp?A&o_PGFvDaTa=*^NEpToA0PQ8K79Bi+RID$zJJqD%D*M1w&b2Y zaDZxE`1G?k=Dt1F+6&+VC>2aSwtviaIt6FW{(RT>?VND$p8X6C>=+)Oj9%CPh@#}} z=bw8kId$q3BId3Elv1?Y4PJTqgL=I-ba3MhM5MaB+~C!T3C73A36#R!IM#K4eZGTO zr_)8OY})|PW%kY&U%h$!-<*E41wmwuDc@r=9jD%{HOI!D8IGeE5r+|nh(#Bcq7;f; zC<|dVGpzOuS*FP{K@1qNSaFC2Z8DPTy8+AwJZgtqD`~|YaL2esUFRVr1Js!XN=x;Q0ZSs2^i2A{wO>MUj;vc5w?pS<2-+FGU53q5$8I@co3*oloQY z6)w*GM4GP9Y|V1z>IHC$FsP!-E?Sp3K&gm4Uo}=(+5%w3HF}-2h{FDAB_6@^LOiM{ zKc?9_kB7pD!5G1arLVS~O1up#3#4fea40{Z*S#jzx~-3FEse57;%JCGZ(@wb_cUqN zWT`bxX#`_R@InUrceB)*W1wzHS62|L*96a_n|2U;y9z9gmG(6UAXXz*DD(Hq)$8FWQhbW?9;ba!ELWdL_~ hcP?peYja~^aAhuUa%Y?FJQ@H1002ovPDHLkV1hDmP>BEl literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/user-desktop.png b/Media/Themes/Umami/Icon/places/user-desktop.png new file mode 100644 index 0000000000000000000000000000000000000000..1748a80a24f235493c8f61f3465e68b7c3027f14 GIT binary patch literal 1109 zcmV-b1giUqP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00S0DL_t(Y$L*C(XjE4e z$A2^PG-|CzjncZbDs*8tDpGA5gbHqK(HJcizp5n0g=k$UWaqj?6k@wbaV0KVsoa2h!vA9f z)C)gkJg;fnwk_m`M*w)>-(}X6?7-_n?;BKFTE5|F{}984B2W4r(tWLa8887T0LF72 zMv5i+2Z!-}~I)rL~Qo|L#*5DTUj|10w)lZB%|P zho8Mk`@Uv+`?7ITRWL>n5kw3|3}S>`J3i*{k3XYVmg8TRG;Ad~L|Jzm7 z6;nk-Pz?zL6+u)GQFd-$%#qHcjGHtwk;Zg52!is7t}B<=xBKhFCbd8kg4(%SH6p_D zrPIhi_bKEDI^Bkd2yXmwa>Mk=k7#J9uL6%ERwwds9uljN&L|D1aG|u5nbdjK-MA9oHCeD3W zXsg87ARMd0j1(+tY&!mAXf_k;-a)X^RS-3ff=a|HCa#H^NWruGd(8jnlh52XjR?}| z^t$Htn`zr~Fzyy{qjg#{N}pE7DqQ_%kguB8Q9tAT4yz(nTLa2Lz>1ZtIdtTl1=hIG zF0dst#-K(xc_z!ww!Pf;heD`vAwi-B#d4WtjbC!G<9C|Bm`f(3QPGPpUXUV1sazrm zDm`)G&&O=q+{%^fJ+-M%666a-o)6>!nDfCrjvoJ)nNxg5vwX3h&sVIa%Yyel?oaHA z2t-WJ*)yj;bUzmI!$19fvHKYC)V7}IoELAwl9jE$oj$*#D$V864V;tgUKEpnY4#n~ z#!$HrM!OR{D!=9lK*sUu#s02cf|LbLTr8uk)%rnvz>+uGD-?|d{B^!#^?NgW3HIkP zk1ltdiU0rrHFQN-bVF}#ZDnqB04QTAATls8G$2Z0Yjt8EQ*>o%Ze?-`3PW;bVRU6= zAa`kWXdqN*WgtgMO;C{8i*En`03~!qSaf7zbY(hYa%Ew3WdJfTGB7PLIV~_YR53U@ zGc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPLFg7hQH&ih)Ix;sp bF*qwQFgh?WLn#C500000NkvXXu0mjf=dR}D literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/user-home.png b/Media/Themes/Umami/Icon/places/user-home.png new file mode 100644 index 0000000000000000000000000000000000000000..9a834163f798fcaf8e05cfe124c69d9e21be60da GIT binary patch literal 1114 zcmV-g1f~0lP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00TEkL_t(Y$JLd+Z(CIu zho5tNZ6{9BI?#aH2&vGj3k(PWiiA)_XGSvmA7IQ7q%36290`#zduLWuDl!BlVBkwZ zD5(_MVkdFz_+H<8zFr2~*RJDGHDcj69qF9&d)}{mF8tr628-q8<-+3P;@m|CtJP|I zeSJM#8nCvu_N!9r79f)jh7b-#O0m1U+uhsS zTd!8Do1+E?2L}LbZ*Qmn3RqcL3AI{H9UdOyoFjxl2q6bVQN+&9PS;vn8r%_)(I6gf zZf@Qi2SkH~_g+E>$~lMk{uzLXV2ojDX=!?Xex8|`8A_!RN-4C~gb>);+PXU_gB;+T z73VD8dyE|dMj@GTw9jIc>LsqV&LNp`YWdSIW*bSxJ-GSE&>x5#V2s6D$NrJy{*Ml= zb-#jzem#a!$=Do-1f=+p!l_eA@cxrKjqi{fpkyQiC*FJNXMqpjxyi>L+(AZsl;T2- zY!wpY`25SS7eL9m9ysSnGGSr5Oud!z*GYG>q2d|%KvTs6*XD}og6>S{ffMH(v4uBg z${-MFea_^yYZdOjwZMKoX6N@~d>E2~KxRBz>DD=bwY?@MR&k|LMg*e55KxXZ@7|v0 z)~hq16t}O>@zPwGhrcvvcO7x+m@XAC#-TLLbAVonJc)tXO0l=u_kZPTnYUk`XR4^_ zq%Mb+BE{V|7ufmjh*oTvE)_|;2H3WoTF<5DVE_OC literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/user-trash.png b/Media/Themes/Umami/Icon/places/user-trash.png new file mode 100644 index 0000000000000000000000000000000000000000..ae881de70b6e5e2a4a911e462eb88d14c62ca846 GIT binary patch literal 1323 zcmV+`1=RY9P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00an0L_t(Y$JLe3Yh6VY z$3OSZ{qgR5FTa}9{vgyUT67^+(S;xgR*BV?r@NWg`HxX@MqfEJ2Bu&AJJx~O2? z73)si7h1s@Z4-)V(zNEim%P6BZtjnnGvlI3nl=?`#f1lkIh@1Hch1arz6bvA=B+#9 zlhJ6kI`H}G>a`a)$N$l0o-c7+J(}me_fMWEH9)YQOp?A&o_PGFvDaTa=*^NEpToA0PQ8K79Bi+RID$zJJqD%D*M1w&b2Y zaDZxE`1G?k=Dt1F+6&+VC>2aSwtviaIt6FW{(RT>?VND$p8X6C>=+)Oj9%CPh@#}} z=bw8kId$q3BId3Elv1?Y4PJTqgL=I-ba3MhM5MaB+~C!T3C73A36#R!IM#K4eZGTO zr_)8OY})|PW%kY&U%h$!-<*E41wmwuDc@r=9jD%{HOI!D8IGeE5r+|nh(#Bcq7;f; zC<|dVGpzOuS*FP{K@1qNSaFC2Z8DPTy8+AwJZgtqD`~|YaL2esUFRVr1Js!XN=x;Q0ZSs2^i2A{wO>MUj;vc5w?pS<2-+FGU53q5$8I@co3*oloQY z6)w*GM4GP9Y|V1z>IHC$FsP!-E?Sp3K&gm4Uo}=(+5%w3HF}-2h{FDAB_6@^LOiM{ zKc?9_kB7pD!5G1arLVS~O1up#3#4fea40{Z*S#jzx~-3FEse57;%JCGZ(@wb_cUqN zWT`bxX#`_R@InUrceB)*W1wzHS62|L*96a_n|2U;y9z9gmG(6UAXXz*DD(Hq)$8FWQhbW?9;ba!ELWdL_~ hcP?peYja~^aAhuUa%Y?FJQ@H1002ovPDHLkV1hDmP>BEl literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/places/xfce-trash_empty.png b/Media/Themes/Umami/Icon/places/xfce-trash_empty.png new file mode 100644 index 0000000000000000000000000000000000000000..ae881de70b6e5e2a4a911e462eb88d14c62ca846 GIT binary patch literal 1323 zcmV+`1=RY9P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00an0L_t(Y$JLe3Yh6VY z$3OSZ{qgR5FTa}9{vgyUT67^+(S;xgR*BV?r@NWg`HxX@MqfEJ2Bu&AJJx~O2? z73)si7h1s@Z4-)V(zNEim%P6BZtjnnGvlI3nl=?`#f1lkIh@1Hch1arz6bvA=B+#9 zlhJ6kI`H}G>a`a)$N$l0o-c7+J(}me_fMWEH9)YQOp?A&o_PGFvDaTa=*^NEpToA0PQ8K79Bi+RID$zJJqD%D*M1w&b2Y zaDZxE`1G?k=Dt1F+6&+VC>2aSwtviaIt6FW{(RT>?VND$p8X6C>=+)Oj9%CPh@#}} z=bw8kId$q3BId3Elv1?Y4PJTqgL=I-ba3MhM5MaB+~C!T3C73A36#R!IM#K4eZGTO zr_)8OY})|PW%kY&U%h$!-<*E41wmwuDc@r=9jD%{HOI!D8IGeE5r+|nh(#Bcq7;f; zC<|dVGpzOuS*FP{K@1qNSaFC2Z8DPTy8+AwJZgtqD`~|YaL2esUFRVr1Js!XN=x;Q0ZSs2^i2A{wO>MUj;vc5w?pS<2-+FGU53q5$8I@co3*oloQY z6)w*GM4GP9Y|V1z>IHC$FsP!-E?Sp3K&gm4Uo}=(+5%w3HF}-2h{FDAB_6@^LOiM{ zKc?9_kB7pD!5G1arLVS~O1up#3#4fea40{Z*S#jzx~-3FEse57;%JCGZ(@wb_cUqN zWT`bxX#`_R@InUrceB)*W1wzHS62|L*96a_n|2U;y9z9gmG(6UAXXz*DD(Hq)$8FWQhbW?9;ba!ELWdL_~ hcP?peYja~^aAhuUa%Y?FJQ@H1002ovPDHLkV1hDmP>BEl literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/audio-volume-high.png b/Media/Themes/Umami/Icon/status/audio-volume-high.png new file mode 100644 index 0000000000000000000000000000000000000000..79c5401ad35c0a75cf6d7255864d350ae09ac187 GIT binary patch literal 1161 zcmV;41a|w0P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00U-8L_t(Y$L*C(Ok7nI z$A9O&Hx85lmQrG+SQqXZTU#J*YGQP$F0_SWCtZ+&+N7GQw2_*Sm*_%G3_%wK(Zou} z2vZtOiW`lx>W56Qm>5jemP8?GR5TrAnEO7Ci}_#*L!@1~@FX|y+_eG z{w3`77WVeN{pOat?>f74$Br9QQ{PRjcb&DKBG+{{c*eZk_0qA*K!evy)ighQ`q9Q` z1|qiwn3?&h1Hs<9wSgcA%SzQ~$RSHBdTtpIJNWdgk?0+7*!CK?>ADtxvBRUPbnNbk zHx^Gj#huo+eMUqCAPhr|Mxz5&Rn_fBUwFR!+~;E@_iz05zFz4VI+yG}_(fA^{6$Oj z!|g32jo&9)%_;?|8gYAE$L(<)AW~OX*M8!a?uwe4+O;KGdJ2)(_8g9HOb*6A0rjHQ zOM7lo(6WDTHEl;LO0;ot*l%|p% z0L;$)Qpm5(Tm``MOz!IyKby-c@(4gcFfCX|00`}G)e5@T1f&2a8ITwr{R=1u z$Ye5jp27Eh&YVg79fr&18e{SvGcg#ujO&IAbF)Vjot+qreR==k_(oJ8azxIrQji!P zT>?1Se=>OI-M#=|Zf^F|k;KU2hVlv&#q+)V{>u@{HN-jr5cRbWk@%&7*pDSW`0e^1 zm8n$9HpW=K?_1N;(@QP;_s$QU{-|@?z1zw>;}xE1K9cXpJ_Bez^zjplJ{YNpJWxDe z3~^iu=lW6v4gfEg-xgryc=23b;eRjw0j92hTU?Le&;S4cC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9w-8h literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/audio-volume-low.png b/Media/Themes/Umami/Icon/status/audio-volume-low.png new file mode 100644 index 0000000000000000000000000000000000000000..35ff977672b2dde8788a5b8ce67ce8ddd8a05dcf GIT binary patch literal 888 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 E2Zv3>xc~qF literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/battery-caution.png b/Media/Themes/Umami/Icon/status/battery-caution.png new file mode 100644 index 0000000000000000000000000000000000000000..9f1da9373bd9ed72c6ac4ef3d4ee0570a6a2b1da GIT binary patch literal 1244 zcmV<21S9*2P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Z7hL_t(Y$F-GzXd7i1 z$3J(Mq;1mGEd4cW(ITC0Zhx&7(c+KI4iN>@jdiVUK>foi3}!`895@vGw-q}C5n-&= z&Z-56;IN2mH#+CKR)>}`C+hlxZnkODCb>&4_rCsdmyk9Aqv8uMclX}=KA-RNeID-M ze;!I9n|j;TH-R@cEWgz~D31PXKySPHBoG2x-5%c$k3IFwUDewkEise2oShujzZp3- zt4Om9=K^gS4DSH(~`TB6*6oh|*WQ2)thbY$;sVetlCgNPWGRLA-$-!q| z0o9FF-oUX#_aLPVzI;-CksVLXAu7uL=0l&u;uIt!0NlH~l}+3BaeDFwUtNx2cRBqL~le#e6^e1@VZPj8Ip%9B;s-F4Mqx*!_QELK!)VSlJQucgvmN{!zQ&R7P?pn04=lVGkvVi{?K5&9AU#;nfl(QAZfJlI3ui zNF;&~A}7nh$sqtbqy)fsVj%s>>p-;Z8mK@K*tL~_loH)AFcL-%!0o8rJu<@R=qOD? zHmvsd>j0%rEHxY8cF1LRo7$>YR+d>^jpV2=Ier3wn&xKu`ueClZ~%aY3m0)jx;6qd zpx8+b=>T*+o)cV*oduw-qXU5H>FIUPhoL=f&qj+fA3m?wC%j%C0ES_#16ar21E6WZ z_PoEpf8BER4XJ-O=-BDibcJn#lvge>> z4?JH${7@$hYy@P3RP0tuPtCcetqcHLuzOy03c3bdb!M{QRR8Pv3$4wFz+C^=Hvj+t zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wiUjzCxH6Ch>g($m7#MPMb0E5dgM%RqGcz*?qp77Hg6iw*>gsAM zD=W&%O6%+EYHMq3VwX7PZgR}s;*`9?rEr@|;dYmzZEj_|+{$*lm+kT{-R@hlE4J-W zV)v2kiuU{|C-bMCESPe#VCu=D>1T>&oGzMirgZkX@_85P>+2dD8=6|`Crp?yckY}e zOBSzPyJqv|&D*wZJ9_l!i4(_9oH%ju;>F9CFJHNGTOWy7}V8 zi`TDTzyA3B&6_uGKK*$6`Nz92KR$f;@afyH&!0bs{_ow%z`&qZ666=mz{JeL&dDzz zA}S^>AuT7bq^zo~ZD?U(X%iL^S37;p{sSj&{rvS(;nN%j1_s6?Z+91l4pvzY1_p+& zo-U3d8WUS5z4SZeAmY~Ta5E$(aLa-PO3Au<8dtW(t=pu`ap}~rfA1ZQE<4@!{PM?k z=AF-~${ha-8>KhpHOiiT$ZmT`OTaDhnJD+7u=uN8QHpkk%K5X_z5KN}-lro`;=KQM zSEH}p-#_iuIbEL86!W}$kcGC>nYF6A!}`_QT%ak}Ti{uLc=+xh)3zr6A>&Sn}%e&=pa!w=WDM=~zo z5wYdB>xZ8V3=DQ9t`Q~9`MJ5Nc_j?aMX8A;sVNHOnI#ztAsML(?w-B@TwEH-nhHh+ z2F40riP@z|3hG&zxw(nS>I%UnshN4HMe4@yBN!MMR6!=CB$lLFC4-G&FfuSQ&^55s zH82Y?w6HQZwK6o*HZZURVeP?Gg@^>bP0l+XkKW;9}G literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/connect_established.png b/Media/Themes/Umami/Icon/status/connect_established.png new file mode 100644 index 0000000000000000000000000000000000000000..501d3f6fc92c962e5aa75e371f3144b4cfef6dd4 GIT binary patch literal 901 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wiUjzCxH6Ch>g($m7#MPMb0E5dgM%RqGcz*?qp77Hg6iw*>gsAM zD=W&%O6%+EYHMrEJSyDcy4~V>Jd^r8lO}j3_xq$z_DP@On?5-(Z9;J7(733eadA=evZ9uyMJ>xp+gFsfuPpCcU0+|<*x1n2Qa@qBgt>F) zELpO6?b(!re{ISdR8j7i?^F3}}>rkXG?FnsWI zaSYLz*gENE&>;r_*JdXM*8oP*$h&TxjNvSt7cwXKin%%&{Q2J=U^;W6N^tA*uQp$H zhU$9$pLm?ta`}OUCx1FlH{d*8)GfMl@7`-#SG&R^p6p$B)%5K_?P*JH=GfTYyEid@ zeTMCQ?XDmdZxO5bkgYL~4qf$J?IU8na{huFnUh7Wc^?-W%vgNGa`6KH7byh`SE*i` zd2&f2$L_kaImwG;cn(S2c(Q)KtnidS%NHCr6tz75Nr$1vXFqpNyJ?ifn$34F|33ER zme^)q4u-jP-w%oT_3@ospQk4r(&9AnN2FHiB%3?j@7c?qWcYl`t)6ZAKrNEng)jRo z0|SFyiEBiObAE1aYF-J0b5UwyNotBhd1gt5LP$ocg1e`002h}=vZjKOfq}7tS7LT) zl7f0xW^Qg`vbsWWNor@#v7>o>z40H`Fbq&ly3@xmT zO|1;gv<(cb3=F1ywlGA|kei>9nO4bQWMF8hYhb2pXdYr{WMyP-WoQY}@HCDi4wPCw MUHx3vIVCg!0H)njU;qFB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/connect_no.png b/Media/Themes/Umami/Icon/status/connect_no.png new file mode 100644 index 0000000000000000000000000000000000000000..9d2c7b7524e592ca36eb8997b1c215d1ec6a1c69 GIT binary patch literal 753 zcma))T}V>_0ELg0VT!fNpUkj}kTYa>@2sUaOJ`1ZbBfLpYBk;F)H$YW-lYZUoL8(B z%v$s>K~u@JAMqu%2OnCE$_GP?nLUJsmbwj=FJTy~r`|g7!Fl=4)45b!lph(sI~+hH zBh8fq*cBvfBY+PBlWzcQsudTCfnj^Znkp0kp^?f9f%8d#g#&&9uq*;*68K&P(6@kC zjW_n%tV08A9i6cYSjtXx|!(1T#Ha&01MOP=y_p7IK#wlnH( zMq`)NI%Kui3u_%EHQl8;$7$0YRdbKNrK{26Y;1QLEly+m17o|>)X{I^`b=EEx$B|1 zYoOUSXyw|j+6rRS0e%JVC6-n*bo&R6T2%hg-XxJywKWlbLjRN^$7qW>Vo{-tfLyz z4B#b`D^?_5?X-KgT1g(Feer{dE%TkvVyAUlJX< zXUoFo{bC+pb~&~pHCErGt&v<5oz&~g=ZC_&4ZePda_^B5byj@+9Z`0Ow5RFx-ndh} zv@SYi)Ny{>m!iA^?~h-O`8l!og`7w=Cy>rb?nykuKO1*Tu)v#Wh;rF&Zx^=OZOPjo z8P7$G4}Y%gTu5*43PY kkfbO9DH2eJ6qJw_9-^s?zrt5#m@@b@MpBgfB&W*y8^gs9H~;_u literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/dialog-error.png b/Media/Themes/Umami/Icon/status/dialog-error.png new file mode 100644 index 0000000000000000000000000000000000000000..f01f85e2f3d3c2a4300681edf5eb5eb3892a985e GIT binary patch literal 1075 zcmV-31kC%1P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00T8iL_t(Y$L*9oXd7o3 z$A8k@>B}jJEn#6PM5fILEf|q0_0Ur25E}29Ofq%J(6JC`q0mBy?ny{jXYEjkmt@Nj z2(*O6i3L%yQ;@7+%huVIbdv6#r^CIc$RbfbhE9Fp9AJsVN@ayNA2G`_nRf|0-YBt8PA% z&;PP`@gnxahe)lC4*IAnDvGF}zK`rSv0RtJ;v&1X+UHB2SN{$EoT$LW{O>O>UCO4` z*O7w*R0JZ&YQCbNimITgG`-S1_wU|m0w>0E5+4IzIaw}eGj0pr-$zAI6+{J5MGOq{ zFQQ1hP3pkq`22j<6aG3a0vZco!P=1rrHjfJvLuB zlFK5GJXD0-vuC`?HuHJ3*CR9a0fePIQNgrr+e7(pW05+`ibL2Z69Ibz&okvHZv_wu z^U?P_2%B(Xh#v1Z8ee1*HsX0~l}d<#sVK_J*bj50idq)`v|12$`0Eu2tk$blVufPp zL=;qvBU3eMK+(DgRxZbOV+(;*#!7z_5Ua!W%}S-&Nv9#5Ml}fYp)YTY$S_ALiYAkE zQYjjZ7tJc%Fn}Q>>f{&@zYBT++|JI<(%iXo$ad3ou?#yodM;FzSTaHU!;h#uS))1d zuYL=E0BXh|K-~N%euwqXJTKX<*FT;9D#bY>h(I^jg1Ad?PQ`@AXzHm6pBbLOV@U&?d(vgRu3Nd{>>lY z7T^N|^Yx=Dj*Tfu5xz%x;KVVQy$oN?L-{Ba9LV&c+k(bl@Z=V(x=;hQfhN!b+CU!& zMgcJcaSY@rPymjZdBW^D=3GGdsJW*0_kew~);9`d91ua`KoXb&(xz8zvmTufjH3b2 z1H6&&1S5GkRI}r1qOe8tQPZ*Ek0dzS|8MMH@ip_qGX;z_0000bbVXQnWMOn=I%9HW zVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 t05UK!F)c7QEipG#F)}(bH##vnD=;uRFfip_2wngH002ovPDHLkV1mU^)#(5L literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/dialog-information.png b/Media/Themes/Umami/Icon/status/dialog-information.png new file mode 100644 index 0000000000000000000000000000000000000000..037bfb801e11d365b0cd07c9ed2e8a28c1ca2cdc GIT binary patch literal 1352 zcmV-O1-JT%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/edittrash.png b/Media/Themes/Umami/Icon/status/edittrash.png new file mode 100644 index 0000000000000000000000000000000000000000..a6b9cd3b5498f66c6a0caa0481b70399a2036e91 GIT binary patch literal 1589 zcmV-52Fm$~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00k0BL_t(Y$F-GdY*bYg z$A52~c{9_!6d@I{g$hLxLlh7dQ9(e7F$Oo{7h_B`Bx>SEzxmz7#2Dj(i62l>O$Za9CUbyJgL~CtNo}vG-^+HdZ@J?&^*l<$iDECGeZH4SeT6|i zrr{|kGx=H&R9@WQ6UB*Ndj`Ps>RxSa=vw*s&gW@wZzq?{a^L+AaP*f&EM2mc-r})z zpJ`dxwnGqx%$wK7!UfB5-IZJ%97^8Jk<#g1A)o&W(e7lwHGTV5!!X{kjFtBKb+^*h z*$G;cah%j)cXu~!^XJpt)WWjv>ycU$g&|T(8XKFbl*`p@Zz(_wf~}q_cAgYu2vCvTd?n zj=S&K!pRfIak9M)vj9@c+`U`YGrz4nrEeJqdM5u`r&2JqzRaelZJ&I4h+^+CmUXWn ziXwF4)UgPoTHP!_9MyKc@%EE@q!d@I^NhKkdp2*QTJ;Hn8sC2XC6#KQ4&}rPm&x?0 z=klXlw{B?3xEX}h$Ycsd6iR|n5=AOWAhpSuNnbEo69hGAZrH+Yw{PUf9}aWw+#i(7 zr^z?|)W7@nv89*E6pZmWqQ&zp98=;r3ARZlYvKtT5i7KIG00L2e1cky(Gi2TmJSYo z_YFhC-;%43$v2EKUI_ZH&e-OH9K?YB_DwX$rK`Oce#%&@9MiC;$G-CYPD8r>9 zQQ}xcU^q0^FJ!oMhK7Q|pBzW3I2A;q`ieh+RR6|6F*07hq|=0}7^x&-JkD_G7)?z} zsa8sa!8sPUufs50yzD$gEEpK|`NvN`Stl*F0MxG7`_%A~k^&~Cpa1{(6MBoS^_w}4&Hw-aC3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9bV6J literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/error.png b/Media/Themes/Umami/Icon/status/error.png new file mode 100644 index 0000000000000000000000000000000000000000..f01f85e2f3d3c2a4300681edf5eb5eb3892a985e GIT binary patch literal 1075 zcmV-31kC%1P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00T8iL_t(Y$L*9oXd7o3 z$A8k@>B}jJEn#6PM5fILEf|q0_0Ur25E}29Ofq%J(6JC`q0mBy?ny{jXYEjkmt@Nj z2(*O6i3L%yQ;@7+%huVIbdv6#r^CIc$RbfbhE9Fp9AJsVN@ayNA2G`_nRf|0-YBt8PA% z&;PP`@gnxahe)lC4*IAnDvGF}zK`rSv0RtJ;v&1X+UHB2SN{$EoT$LW{O>O>UCO4` z*O7w*R0JZ&YQCbNimITgG`-S1_wU|m0w>0E5+4IzIaw}eGj0pr-$zAI6+{J5MGOq{ zFQQ1hP3pkq`22j<6aG3a0vZco!P=1rrHjfJvLuB zlFK5GJXD0-vuC`?HuHJ3*CR9a0fePIQNgrr+e7(pW05+`ibL2Z69Ibz&okvHZv_wu z^U?P_2%B(Xh#v1Z8ee1*HsX0~l}d<#sVK_J*bj50idq)`v|12$`0Eu2tk$blVufPp zL=;qvBU3eMK+(DgRxZbOV+(;*#!7z_5Ua!W%}S-&Nv9#5Ml}fYp)YTY$S_ALiYAkE zQYjjZ7tJc%Fn}Q>>f{&@zYBT++|JI<(%iXo$ad3ou?#yodM;FzSTaHU!;h#uS))1d zuYL=E0BXh|K-~N%euwqXJTKX<*FT;9D#bY>h(I^jg1Ad?PQ`@AXzHm6pBbLOV@U&?d(vgRu3Nd{>>lY z7T^N|^Yx=Dj*Tfu5xz%x;KVVQy$oN?L-{Ba9LV&c+k(bl@Z=V(x=;hQfhN!b+CU!& zMgcJcaSY@rPymjZdBW^D=3GGdsJW*0_kew~);9`d91ua`KoXb&(xz8zvmTufjH3b2 z1H6&&1S5GkRI}r1qOe8tQPZ*Ek0dzS|8MMH@ip_qGX;z_0000bbVXQnWMOn=I%9HW zVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 t05UK!F)c7QEipG#F)}(bH##vnD=;uRFfip_2wngH002ovPDHLkV1mU^)#(5L literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/folder-drag-accept.png b/Media/Themes/Umami/Icon/status/folder-drag-accept.png new file mode 100644 index 0000000000000000000000000000000000000000..0c7a7039fe43990fc50a85059c956cd3bbb42eed GIT binary patch literal 1041 zcmV+s1n&EZP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00QnwL_t(Y$JLX+ZyZ$+ z$3OFKbM|pKIfH3%h3F}eBDq6?hJp@R2#G6bkZ4`FaJ2er_ru=z-VDXwp6?E?0~QUhnq9p&pZR`g2LA6yiePnhwX(dtyij(r z(P(tjG}S8uHa0fyi^!XRU35`ZkId}m=H_PmN`UqCb!#@8#27I%RP$oZ3=v^(Z*O>T zaF8|{jjhQ9tyT+wot>T2O95+ZYuao!<>=^$5CW=-s*V9klCZnGJM`XHi#KLAS;XC~ zt*v)w0Z9?W7)@0rgg}gO3SeerS;orBN^Nm*k$SyOwOU0)aL%EsY;SMBbCH5EAOvqA zcw&rGfJsOQfquV_NG{wtH-=i~F@yd-F?OygmKvW|O z%4rtKg7WIZ9FL#$(5Mfm0D!9I$sTp8EG;$Y^aCO04tq6Bji~upMl-xOg4wt;P9U1` zuyxW47H`i2h*3FyF8tcA^8Q;(yj-hLO#~kl#V%DMK*c!@k2?&{vS&Zs`s6k=$ACmc z?3dpMeAil_ext^{2al;#WM(waKPLYdVGPP2C!OKxp!+F^O)8ih305UK#FfA}SEig7zF*rIi zH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-Lg zIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000 LNkvXXu0mjfELXLc literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/folder-open.png b/Media/Themes/Umami/Icon/status/folder-open.png new file mode 100644 index 0000000000000000000000000000000000000000..def61fe71f94ef2eb82d10d3159a4ff1e5e69707 GIT binary patch literal 1073 zcmV-11kU@3P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Rz5L_t(Y$JLd+Yh+av zz<=k?o0%kI+!+=dqlF*;!a^Dmto#E^If7szmMK!Cnnp++5F#k1vaz;z16hdJ$P!(% z;D$AuRr9sUWMyam`*2UT^?%F4?F+V>)-Dvul=Iss_(dG`JKN9kAuWx%)h>r|?BM8GUxU=$3y=$>L z0(hSnxd8Ms`0V3%s5pm}`xyfPx!5Tjki6-q1mUbVr1RC?<`pR_wI}(-U-!G$n%F|> zRY&jvC+=_z;JqJBag2%+p*si+vv_UeQH{Y+=$`ix*uNZsh7d3_o~6R_&d+|jxf?7@>h8;@V6ajV8p51&w}$k^*p zp5^k;6Fazu*7v?)`%gLSrU5Jv!s`+PzbVn3?(Ghxa~zwDHA0 z`Dj>?!7A`7@XF=1{Yz2c4A=!ycs`&4R4$CmO)QtPrGRI`)BB&x-yN?rIq#q_-~a#s zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf1@qhs literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/folder-visiting.png b/Media/Themes/Umami/Icon/status/folder-visiting.png new file mode 100644 index 0000000000000000000000000000000000000000..8d6b6e61324de102c53c1d1f58d41630e255bf35 GIT binary patch literal 792 zcmY+7ZBWYr0Ed5-c15wKq~s)rit_(2TAQ5KHEq(XWW~~I|5}{2wz1gJ2#2U;i^%1D zog-0fUZN9s`Y?98a2b&>xl8*Yb^;#HOH{Hv3${* z3!mYAKs3@jyRCwH6UeZl|FW<|KX*)@ze}fUY6_Je^9ka&S8x-@I$o${zK7S%zcIB_ z@}O#kf0tiZ(UX~)@XxIYlBusuR6~X)#bhf!*_>2=Zh_0l$kx_~uSWXrgpuE?jtudd zUydq$wZ{{$KeL+6r>bQE9<<7`oS_vV!}p|jZPJIl;yJ8EOW8P`-PkB7yjN~_v`7aA z2Y1%>YxfJ+U!><`PBWS6vUr;;hxB^;TIEc-S3*cCRbzB-SYk+fx3Gt!GlrZu1S_xd zv-+QI>GHvQ3XvWz`^2x=Kh8T#GW&A1hNiPF#$d(_)8<;xC*1GOxa5;8QeaMU6ry68d}`XldE$+D1I=?Y%#Ck>O=qULD-{<-4Pe~mAf|noUMUT%`LE7Ji5P~F&vDxYRcE=;^zC?pXz1iU;f&KD~Woq zFP>$$x(7=g!`oYnJ1tR~mm5)s5<^)=6-`d-@s9C$eSH004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Rz5L_t(Y$JLd+Yh+av zz<=k?o0%kI+!+=dqlF*;!a^Dmto#E^If7szmMK!Cnnp++5F#k1vaz;z16hdJ$P!(% z;D$AuRr9sUWMyam`*2UT^?%F4?F+V>)-Dvul=Iss_(dG`JKN9kAuWx%)h>r|?BM8GUxU=$3y=$>L z0(hSnxd8Ms`0V3%s5pm}`xyfPx!5Tjki6-q1mUbVr1RC?<`pR_wI}(-U-!G$n%F|> zRY&jvC+=_z;JqJBag2%+p*si+vv_UeQH{Y+=$`ix*uNZsh7d3_o~6R_&d+|jxf?7@>h8;@V6ajV8p51&w}$k^*p zp5^k;6Fazu*7v?)`%gLSrU5Jv!s`+PzbVn3?(Ghxa~zwDHA0 z`Dj>?!7A`7@XF=1{Yz2c4A=!ycs`&4R4$CmO)QtPrGRI`)BB&x-yN?rIq#q_-~a#s zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf1@qhs literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-dev-wavelan-encrypted.png b/Media/Themes/Umami/Icon/status/gnome-dev-wavelan-encrypted.png new file mode 100644 index 0000000000000000000000000000000000000000..28fe5ef49659ce5a7251fa41d46b8aa1695cedbf GIT binary patch literal 1309 zcmV+&1>*XNP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00bXNL_t(Y$K_UEOj~sn zo|_t@4;p<_E2*oS|!1qKTxaNF)!1-0K{Y*gG2orjjFm2|IV7jEEh@7FU;MoKqip2pU%vLi%2-qTlo4UguGm^E(a0^ z4j|^RDe`n*?h@~1(+2W=DneE?+lpFzMogT*Bu?IjU>}<(s|*Z@QrHTSso*H@{}R|K2fro004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00QnwL_t(Y$JLX+ZyZ$+ z$3OFKbM|pKIfH3%h3F}eBDq6?hJp@R2#G6bkZ4`FaJ2er_ru=z-VDXwp6?E?0~QUhnq9p&pZR`g2LA6yiePnhwX(dtyij(r z(P(tjG}S8uHa0fyi^!XRU35`ZkId}m=H_PmN`UqCb!#@8#27I%RP$oZ3=v^(Z*O>T zaF8|{jjhQ9tyT+wot>T2O95+ZYuao!<>=^$5CW=-s*V9klCZnGJM`XHi#KLAS;XC~ zt*v)w0Z9?W7)@0rgg}gO3SeerS;orBN^Nm*k$SyOwOU0)aL%EsY;SMBbCH5EAOvqA zcw&rGfJsOQfquV_NG{wtH-=i~F@yd-F?OygmKvW|O z%4rtKg7WIZ9FL#$(5Mfm0D!9I$sTp8EG;$Y^aCO04tq6Bji~upMl-xOg4wt;P9U1` zuyxW47H`i2h*3FyF8tcA^8Q;(yj-hLO#~kl#V%DMK*c!@k2?&{vS&Zs`s6k=$ACmc z?3dpMeAil_ext^{2al;#WM(waKPLYdVGPP2C!OKxp!+F^O)8ih305UK#FfA}SEig7zF*rIi zH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-Lg zIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000 LNkvXXu0mjfELXLc literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-fs-directory-visiting.png b/Media/Themes/Umami/Icon/status/gnome-fs-directory-visiting.png new file mode 100644 index 0000000000000000000000000000000000000000..8d6b6e61324de102c53c1d1f58d41630e255bf35 GIT binary patch literal 792 zcmY+7ZBWYr0Ed5-c15wKq~s)rit_(2TAQ5KHEq(XWW~~I|5}{2wz1gJ2#2U;i^%1D zog-0fUZN9s`Y?98a2b&>xl8*Yb^;#HOH{Hv3${* z3!mYAKs3@jyRCwH6UeZl|FW<|KX*)@ze}fUY6_Je^9ka&S8x-@I$o${zK7S%zcIB_ z@}O#kf0tiZ(UX~)@XxIYlBusuR6~X)#bhf!*_>2=Zh_0l$kx_~uSWXrgpuE?jtudd zUydq$wZ{{$KeL+6r>bQE9<<7`oS_vV!}p|jZPJIl;yJ8EOW8P`-PkB7yjN~_v`7aA z2Y1%>YxfJ+U!><`PBWS6vUr;;hxB^;TIEc-S3*cCRbzB-SYk+fx3Gt!GlrZu1S_xd zv-+QI>GHvQ3XvWz`^2x=Kh8T#GW&A1hNiPF#$d(_)8<;xC*1GOxa5;8QeaMU6ry68d}`XldE$+D1I=?Y%#Ck>O=qULD-{<-4Pe~mAf|noUMUT%`LE7Ji5P~F&vDxYRcE=;^zC?pXz1iU;f&KD~Woq zFP>$$x(7=g!`oYnJ1tR~mm5)s5<^)=6-`d-@s9C$eSH004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00O;9L_t(Y$JLb0YZGA* z#(y)LY?@@ZsaQ?v!Pa^Z1(j-|=t;y&(PQsDh~J8+Eh4C3wNh)Xp0vfkp*O{|h@hY; zil2x2eN+s2X^Tyh^|0GC!9^M>f&=ei-kE)#eP*6r_}@dSlfj{Ze&GC406ovUboR`- z3!%lqp#g8HF*r2fRbxiNh{ngqeR%?#H*Y~iD#yPUa=9GyP1#Ha5uv4}6K}QhemK52UW!w!`PisMn`XP^zgBuKb%0rFMauM-@YQf zGR^5zeE{_I^iU`iP!)Pl^^tTO?%lszV}L~whzhF8_3DthH|+~B9TBu@epQa zW;lHC2#=pU#$QPt1oQJW#?aX4;(29M6&1mn(Iy*ZVyE956 zkw7h9#m$kMhzL>J=Jwbv+S}SG73ZKH0u?CE&a!DkC!aojWO(EnrBaDlLkv4+Q=FUQ z`i)^ee)vF3x|N@^vj|=&LFf*iK7GbqvBB||hCrMbD8_Vza7i8$-lu4iIm0?+fXZJSJH8yy{MSdm_V9kuIp zQQ(SLEJkZZMt<^mieEWh>%J}aa@P)?rubc zh!sJsppMl49V}no%-@O{f^xZh`RcVR=avHa<@48lHRJvRu>-ilaeySgtu4R% zI65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(b dH##vnD=;uRFfip_2wngH002ovPDHLkV1iFdnm_;m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-fs-trash-full.png b/Media/Themes/Umami/Icon/status/gnome-fs-trash-full.png new file mode 100644 index 0000000000000000000000000000000000000000..a6b9cd3b5498f66c6a0caa0481b70399a2036e91 GIT binary patch literal 1589 zcmV-52Fm$~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00k0BL_t(Y$F-GdY*bYg z$A52~c{9_!6d@I{g$hLxLlh7dQ9(e7F$Oo{7h_B`Bx>SEzxmz7#2Dj(i62l>O$Za9CUbyJgL~CtNo}vG-^+HdZ@J?&^*l<$iDECGeZH4SeT6|i zrr{|kGx=H&R9@WQ6UB*Ndj`Ps>RxSa=vw*s&gW@wZzq?{a^L+AaP*f&EM2mc-r})z zpJ`dxwnGqx%$wK7!UfB5-IZJ%97^8Jk<#g1A)o&W(e7lwHGTV5!!X{kjFtBKb+^*h z*$G;cah%j)cXu~!^XJpt)WWjv>ycU$g&|T(8XKFbl*`p@Zz(_wf~}q_cAgYu2vCvTd?n zj=S&K!pRfIak9M)vj9@c+`U`YGrz4nrEeJqdM5u`r&2JqzRaelZJ&I4h+^+CmUXWn ziXwF4)UgPoTHP!_9MyKc@%EE@q!d@I^NhKkdp2*QTJ;Hn8sC2XC6#KQ4&}rPm&x?0 z=klXlw{B?3xEX}h$Ycsd6iR|n5=AOWAhpSuNnbEo69hGAZrH+Yw{PUf9}aWw+#i(7 zr^z?|)W7@nv89*E6pZmWqQ&zp98=;r3ARZlYvKtT5i7KIG00L2e1cky(Gi2TmJSYo z_YFhC-;%43$v2EKUI_ZH&e-OH9K?YB_DwX$rK`Oce#%&@9MiC;$G-CYPD8r>9 zQQ}xcU^q0^FJ!oMhK7Q|pBzW3I2A;q`ieh+RR6|6F*07hq|=0}7^x&-JkD_G7)?z} zsa8sa!8sPUufs50yzD$gEEpK|`NvN`Stl*F0MxG7`_%A~k^&~Cpa1{(6MBoS^_w}4&Hw-aC3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9bV6J literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-netstatus-disconn.png b/Media/Themes/Umami/Icon/status/gnome-netstatus-disconn.png new file mode 100644 index 0000000000000000000000000000000000000000..9d2c7b7524e592ca36eb8997b1c215d1ec6a1c69 GIT binary patch literal 753 zcma))T}V>_0ELg0VT!fNpUkj}kTYa>@2sUaOJ`1ZbBfLpYBk;F)H$YW-lYZUoL8(B z%v$s>K~u@JAMqu%2OnCE$_GP?nLUJsmbwj=FJTy~r`|g7!Fl=4)45b!lph(sI~+hH zBh8fq*cBvfBY+PBlWzcQsudTCfnj^Znkp0kp^?f9f%8d#g#&&9uq*;*68K&P(6@kC zjW_n%tV08A9i6cYSjtXx|!(1T#Ha&01MOP=y_p7IK#wlnH( zMq`)NI%Kui3u_%EHQl8;$7$0YRdbKNrK{26Y;1QLEly+m17o|>)X{I^`b=EEx$B|1 zYoOUSXyw|j+6rRS0e%JVC6-n*bo&R6T2%hg-XxJywKWlbLjRN^$7qW>Vo{-tfLyz z4B#b`D^?_5?X-KgT1g(Feer{dE%TkvVyAUlJX< zXUoFo{bC+pb~&~pHCErGt&v<5oz&~g=ZC_&4ZePda_^B5byj@+9Z`0Ow5RFx-ndh} zv@SYi)Ny{>m!iA^?~h-O`8l!og`7w=Cy>rb?nykuKO1*Tu)v#Wh;rF&Zx^=OZOPjo z8P7$G4}Y%gTu5*43PY kkfbO9DH2eJ6qJw_9-^s?zrt5#m@@b@MpBgfB&W*y8^gs9H~;_u literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-netstatus-error.png b/Media/Themes/Umami/Icon/status/gnome-netstatus-error.png new file mode 100644 index 0000000000000000000000000000000000000000..70249ef7c9b0ce88265f92aec5ac3003d7bc04db GIT binary patch literal 1138 zcmV-&1daQNP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3labT3lag+-G2N400SyXL_t(Y$GwzKXcKu9 z$3K&qsY$F#O;j+n?ov`}yIZxj!WIz^Dhi7iQSlF6t#}Y&7xu7hQM`$u_!m5Q>8ceG z@uJAyMV5+Gjjl~h(zNNcNw8BjY0b>R9w`Yto``@Q!*^Ly|6e(;S=Dd%wd zOwV0lPZc?@{q^UmV}%c`f`C0cw(Z34Zz}7UN+r2{=Vmuh6#&1#iCu@zmi2pl`;U^f ziU2h=H2`$|(h3qJpeXq#5{bf-C$Hv=tq5S-HUJKX3Q8_*Aw58NXnfw-iU2|g08~xQ zuSkrHNL^#}(3^~YiJITgkh-?;H2!s#_+s?gbvrI=K{LtCyR$+fpiF92DU_3f9uYx@07%q3hwlR|?q5Jn)+tR|+yLZTZ@)>}l;B0Ot8VGz6&{tHVcsve()9ECYN)`QiU0q$S z2S<LL>eAd*RhqM&-cxLaB<;&J)GFb09k6sG9(nI1`nrcRiw?!ATA zvNOKMMwcrOPftuR5{U>inH*FsYkOhP3fG`_R!=;@QlKz#7BCH%z(+tX0qr~-birN$ zKLObAHUoW+;oQAk%=fYX0YZX`4@34%o%Ze?;OGw)ac001R) zMObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHYO`e7ytkO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-netstatus-idle.png b/Media/Themes/Umami/Icon/status/gnome-netstatus-idle.png new file mode 100644 index 0000000000000000000000000000000000000000..501d3f6fc92c962e5aa75e371f3144b4cfef6dd4 GIT binary patch literal 901 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wiUjzCxH6Ch>g($m7#MPMb0E5dgM%RqGcz*?qp77Hg6iw*>gsAM zD=W&%O6%+EYHMrEJSyDcy4~V>Jd^r8lO}j3_xq$z_DP@On?5-(Z9;J7(733eadA=evZ9uyMJ>xp+gFsfuPpCcU0+|<*x1n2Qa@qBgt>F) zELpO6?b(!re{ISdR8j7i?^F3}}>rkXG?FnsWI zaSYLz*gENE&>;r_*JdXM*8oP*$h&TxjNvSt7cwXKin%%&{Q2J=U^;W6N^tA*uQp$H zhU$9$pLm?ta`}OUCx1FlH{d*8)GfMl@7`-#SG&R^p6p$B)%5K_?P*JH=GfTYyEid@ zeTMCQ?XDmdZxO5bkgYL~4qf$J?IU8na{huFnUh7Wc^?-W%vgNGa`6KH7byh`SE*i` zd2&f2$L_kaImwG;cn(S2c(Q)KtnidS%NHCr6tz75Nr$1vXFqpNyJ?ifn$34F|33ER zme^)q4u-jP-w%oT_3@ospQk4r(&9AnN2FHiB%3?j@7c?qWcYl`t)6ZAKrNEng)jRo z0|SFyiEBiObAE1aYF-J0b5UwyNotBhd1gt5LP$ocg1e`002h}=vZjKOfq}7tS7LT) zl7f0xW^Qg`vbsWWNor@#v7>o>z40H`Fbq&ly3@xmT zO|1;gv<(cb3=F1ywlGA|kei>9nO4bQWMF8hYhb2pXdYr{WMyP-WoQY}@HCDi4wPCw MUHx3vIVCg!0H)njU;qFB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-netstatus-rx.png b/Media/Themes/Umami/Icon/status/gnome-netstatus-rx.png new file mode 100644 index 0000000000000000000000000000000000000000..d0cdddeb2593b437c24a24c426a1e61abdad5133 GIT binary patch literal 1027 zcmbW0e^8SJ9LB#Jpn!Uo5keupX9{_=dN*>w+>*gyn*t3syk!TujR6ba?d5epz}()A zcA`?k2_lE!b^O%8t|DfQegFqJ(hTCqZj?1WyC@hSLxUaXb+o_wx96Vc?)m5Q{P&y_ zW%0L#z7Prk+k^sM4gf@KBsc&7oY#N&9smN1xS3pl&#Jeq%Ypy^Q3!G}0ZzpLn5_Ul z05H!2=um*ajsvh70CwU{_G~r)gy4d=P6BM%u@TBo_e|>m6s0Ibk_P`vBxwQw3_Ibc z#bU9)Fquq$sj!lMlcdRHGM1N@>Ge92G#QPjnSwHQUKKI9yEEaQxm8_|$X>M+6Yiqr6qr0c4$7Zt)4-Y#Ww;T?~`1tt5#Kh#} z2$eVGc&H)+1a_dxw%Jw&(F`#FRyw$9*=i*VPRomWp#0JacOC3`H9c# z^@=)6UI%~}ZBCXLr~!eSL$*H?_I&uxh!>-G?|EroTpT?yF^PRtns2<^{awFfX3h5} zvfKm!6!{T856%Yd5&%$k33*&eL}bvwH|poO5zz1I`_b?;jZI$-F#y%oS5i1$+6KXWLbHeRJXe9WP^=WRjDw>Z@PZlkzT-LUmg zGC1};?(GSp_7@Kp{?aZ%VC;k`H@n1w_fn&OD{p?%=-Y0cT8VN$j3q4Y^-b=Hr<|{o z$KHE;M1A&=h4y7?>bSnD!SI^&)gmKV`XMqmg&7Zj+}$b2P3}oLnwJ{|f$ZY0jt5<* zIEPGX{=iVQh#hk*a^QYI|L#ty&m-p(Uu|-$hYNl^y_O|?r~Y0G{z|xWe->s44Sds1 z;iTOi`QzfVQGg%(w(-DPyD$&uK1RTw5y#3Ev?Ql(ZB zvGI#i0NA&&R3OtTS&DyWfPpel8k$5ynG$*;EB*kB&WuA*7K+w;6Y2i}FinA~upAgD kolZlUGE2?X<58aQjS;r15lunDgXcg literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-netstatus-tx.png b/Media/Themes/Umami/Icon/status/gnome-netstatus-tx.png new file mode 100644 index 0000000000000000000000000000000000000000..bf2513307db0b25af9bf5c4175d1afa7ebe74246 GIT binary patch literal 1030 zcmbW0Z&1?(9LB%nPi`1x3W{*m4-IB;T)zz&Y@sqXU=v1@?Tj)eWBV~Dj?JIjfF=8F zxDylZSOLNb=;g>D#5dQ=77x%cGd;B^Vw`WW0 zvL*G|dPlZ?AV=@umo)H8tbAERuI_rCqA^d=Bv3R8Djf$(ts-S(5q`et^bKi6hqUfy zS!J!-WK)}M>L!=E*`+brH0DbhvrW_F(lop9QuS4nwcWb5 zA1coFR9v`A5Jt1vRAV8mR%=H`duOMuxA#haf4|*s9~l{OI!B#O=lJ-z+wGp1n3$ZL zoSK^Qcs$e7(=#*Ev$L~vb8~Z#9?#Fu&#$a4EG#T|*A^ES7gyJomX?;6mskGsdA(lY z=ViMAAYNOLFM^=pkPVcVH$_Bl-4+$|+Uq-Z#w8}w($bje2MZfMkiKQ;|B;*?Uz94s144&N;HxRsOIV(cIdcU^&0O(@xI@-Fy8VSKW577RFNEaxcNRi z^%6DLwc5I+=51&f{P?@hz{TQy_D^o5H5qucp6B)cKm1J^?K%A*e=;QlV8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wiUjzCxH6Ch>g($m7#MPMb0E5dgM%RqGcz*?qp77Hg6iw*>gsAM zD=W&%O6%+EYHMq3VwX7PZgR}s;*`9?rEr@|;dYmzZEj_|+{$*lm+kT{-R@hlE4J-W zV)v2kiuU{|C-bMCESPe#VCu=D>1T>&oGzMirgZkX@_85P>+2dD8=6|`Crp?yckY}e zOBSzPyJqv|&D*wZJ9_l!i4(_9oH%ju;>F9CFJHNGTOWy7}V8 zi`TDTzyA3B&6_uGKK*$6`Nz92KR$f;@afyH&!0bs{_ow%z`&qZ666=mz{JeL&dDzz zA}S^>AuT7bq^zo~ZD?U(X%iL^S37;p{sSj&{rvS(;nN%j1_s6?Z+91l4pvzY1_p+& zo-U3d8WUS5z4SZeAmY~Ta5E$(aLa-PO3Au<8dtW(t=pu`ap}~rfA1ZQE<4@!{PM?k z=AF-~${ha-8>KhpHOiiT$ZmT`OTaDhnJD+7u=uN8QHpkk%K5X_z5KN}-lro`;=KQM zSEH}p-#_iuIbEL86!W}$kcGC>nYF6A!}`_QT%ak}Ti{uLc=+xh)3zr6A>&Sn}%e&=pa!w=WDM=~zo z5wYdB>xZ8V3=DQ9t`Q~9`MJ5Nc_j?aMX8A;sVNHOnI#ztAsML(?w-B@TwEH-nhHh+ z2F40riP@z|3hG&zxw(nS>I%UnshN4HMe4@yBN!MMR6!=CB$lLFC4-G&FfuSQ&^55s zH82Y?w6HQZwK6o*HZZURVeP?Gg@^>bP0l+XkKW;9}G literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gnome-stock-trash-full.png b/Media/Themes/Umami/Icon/status/gnome-stock-trash-full.png new file mode 100644 index 0000000000000000000000000000000000000000..a6b9cd3b5498f66c6a0caa0481b70399a2036e91 GIT binary patch literal 1589 zcmV-52Fm$~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00k0BL_t(Y$F-GdY*bYg z$A52~c{9_!6d@I{g$hLxLlh7dQ9(e7F$Oo{7h_B`Bx>SEzxmz7#2Dj(i62l>O$Za9CUbyJgL~CtNo}vG-^+HdZ@J?&^*l<$iDECGeZH4SeT6|i zrr{|kGx=H&R9@WQ6UB*Ndj`Ps>RxSa=vw*s&gW@wZzq?{a^L+AaP*f&EM2mc-r})z zpJ`dxwnGqx%$wK7!UfB5-IZJ%97^8Jk<#g1A)o&W(e7lwHGTV5!!X{kjFtBKb+^*h z*$G;cah%j)cXu~!^XJpt)WWjv>ycU$g&|T(8XKFbl*`p@Zz(_wf~}q_cAgYu2vCvTd?n zj=S&K!pRfIak9M)vj9@c+`U`YGrz4nrEeJqdM5u`r&2JqzRaelZJ&I4h+^+CmUXWn ziXwF4)UgPoTHP!_9MyKc@%EE@q!d@I^NhKkdp2*QTJ;Hn8sC2XC6#KQ4&}rPm&x?0 z=klXlw{B?3xEX}h$Ycsd6iR|n5=AOWAhpSuNnbEo69hGAZrH+Yw{PUf9}aWw+#i(7 zr^z?|)W7@nv89*E6pZmWqQ&zp98=;r3ARZlYvKtT5i7KIG00L2e1cky(Gi2TmJSYo z_YFhC-;%43$v2EKUI_ZH&e-OH9K?YB_DwX$rK`Oce#%&@9MiC;$G-CYPD8r>9 zQQ}xcU^q0^FJ!oMhK7Q|pBzW3I2A;q`ieh+RR6|6F*07hq|=0}7^x&-JkD_G7)?z} zsa8sa!8sPUufs50yzD$gEEpK|`NvN`Stl*F0MxG7`_%A~k^&~Cpa1{(6MBoS^_w}4&Hw-aC3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9bV6J literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gtk-dialog-error.png b/Media/Themes/Umami/Icon/status/gtk-dialog-error.png new file mode 100644 index 0000000000000000000000000000000000000000..f01f85e2f3d3c2a4300681edf5eb5eb3892a985e GIT binary patch literal 1075 zcmV-31kC%1P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00T8iL_t(Y$L*9oXd7o3 z$A8k@>B}jJEn#6PM5fILEf|q0_0Ur25E}29Ofq%J(6JC`q0mBy?ny{jXYEjkmt@Nj z2(*O6i3L%yQ;@7+%huVIbdv6#r^CIc$RbfbhE9Fp9AJsVN@ayNA2G`_nRf|0-YBt8PA% z&;PP`@gnxahe)lC4*IAnDvGF}zK`rSv0RtJ;v&1X+UHB2SN{$EoT$LW{O>O>UCO4` z*O7w*R0JZ&YQCbNimITgG`-S1_wU|m0w>0E5+4IzIaw}eGj0pr-$zAI6+{J5MGOq{ zFQQ1hP3pkq`22j<6aG3a0vZco!P=1rrHjfJvLuB zlFK5GJXD0-vuC`?HuHJ3*CR9a0fePIQNgrr+e7(pW05+`ibL2Z69Ibz&okvHZv_wu z^U?P_2%B(Xh#v1Z8ee1*HsX0~l}d<#sVK_J*bj50idq)`v|12$`0Eu2tk$blVufPp zL=;qvBU3eMK+(DgRxZbOV+(;*#!7z_5Ua!W%}S-&Nv9#5Ml}fYp)YTY$S_ALiYAkE zQYjjZ7tJc%Fn}Q>>f{&@zYBT++|JI<(%iXo$ad3ou?#yodM;FzSTaHU!;h#uS))1d zuYL=E0BXh|K-~N%euwqXJTKX<*FT;9D#bY>h(I^jg1Ad?PQ`@AXzHm6pBbLOV@U&?d(vgRu3Nd{>>lY z7T^N|^Yx=Dj*Tfu5xz%x;KVVQy$oN?L-{Ba9LV&c+k(bl@Z=V(x=;hQfhN!b+CU!& zMgcJcaSY@rPymjZdBW^D=3GGdsJW*0_kew~);9`d91ua`KoXb&(xz8zvmTufjH3b2 z1H6&&1S5GkRI}r1qOe8tQPZ*Ek0dzS|8MMH@ip_qGX;z_0000bbVXQnWMOn=I%9HW zVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu3 t05UK!F)c7QEipG#F)}(bH##vnD=;uRFfip_2wngH002ovPDHLkV1mU^)#(5L literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gtk-dialog-info.png b/Media/Themes/Umami/Icon/status/gtk-dialog-info.png new file mode 100644 index 0000000000000000000000000000000000000000..037bfb801e11d365b0cd07c9ed2e8a28c1ca2cdc GIT binary patch literal 1352 zcmV-O1-JT%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gtk-directory.png b/Media/Themes/Umami/Icon/status/gtk-directory.png new file mode 100644 index 0000000000000000000000000000000000000000..def61fe71f94ef2eb82d10d3159a4ff1e5e69707 GIT binary patch literal 1073 zcmV-11kU@3P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Rz5L_t(Y$JLd+Yh+av zz<=k?o0%kI+!+=dqlF*;!a^Dmto#E^If7szmMK!Cnnp++5F#k1vaz;z16hdJ$P!(% z;D$AuRr9sUWMyam`*2UT^?%F4?F+V>)-Dvul=Iss_(dG`JKN9kAuWx%)h>r|?BM8GUxU=$3y=$>L z0(hSnxd8Ms`0V3%s5pm}`xyfPx!5Tjki6-q1mUbVr1RC?<`pR_wI}(-U-!G$n%F|> zRY&jvC+=_z;JqJBag2%+p*si+vv_UeQH{Y+=$`ix*uNZsh7d3_o~6R_&d+|jxf?7@>h8;@V6ajV8p51&w}$k^*p zp5^k;6Fazu*7v?)`%gLSrU5Jv!s`+PzbVn3?(Ghxa~zwDHA0 z`Dj>?!7A`7@XF=1{Yz2c4A=!ycs`&4R4$CmO)QtPrGRI`)BB&x-yN?rIq#q_-~a#s zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf1@qhs literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/gtk-missing-image.png b/Media/Themes/Umami/Icon/status/gtk-missing-image.png new file mode 100644 index 0000000000000000000000000000000000000000..82dd93df92a7645f2775990829e08a7ac7b152bd GIT binary patch literal 808 zcmV+@1K0eCP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JsWL_t(Y$JLb2YZFlr z$GGXUV{?K@~+y%qupmJ39PDUfubZU6wxd-u_6?7_1vB(}DeQy?KpsnT6!KYjoJ zbRR!~clZlyH?A+IAS?+0+;$uBTQ>m!%5lK!>wz4Jq99DB{pKy~SFhps`^bIYKUUIf z?9Go?f@ifLV8(P!6TSK_hF`uS_3!~wr4qDk7FQ0tp(=$q0)S>+XxS__wqGEsu|xof zFr1+?zL_i#W)Sn(Ck6?FNx;J6@9B*u0fj&VfX$1SBX=Cd0)S4Z^T*fnZ!c>mlQ|t= zSr&4++?hJIENklei-4WRyZXx&O=ib13T*fm=ll3on$gO7>!0|v)Q!66ac^xfbJwN05qjkOew{a zCBE+yGb;d|Qp%qca!)L0TKu2bZ&ilylR`}lnE(I)C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML mHZ3tXR53C-GB-LgIV&(QIxsNhUI<004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00O;9L_t(Y$JLb0YZGA* z#(y)LY?@@ZsaQ?v!Pa^Z1(j-|=t;y&(PQsDh~J8+Eh4C3wNh)Xp0vfkp*O{|h@hY; zil2x2eN+s2X^Tyh^|0GC!9^M>f&=ei-kE)#eP*6r_}@dSlfj{Ze&GC406ovUboR`- z3!%lqp#g8HF*r2fRbxiNh{ngqeR%?#H*Y~iD#yPUa=9GyP1#Ha5uv4}6K}QhemK52UW!w!`PisMn`XP^zgBuKb%0rFMauM-@YQf zGR^5zeE{_I^iU`iP!)Pl^^tTO?%lszV}L~whzhF8_3DthH|+~B9TBu@epQa zW;lHC2#=pU#$QPt1oQJW#?aX4;(29M6&1mn(Iy*ZVyE956 zkw7h9#m$kMhzL>J=Jwbv+S}SG73ZKH0u?CE&a!DkC!aojWO(EnrBaDlLkv4+Q=FUQ z`i)^ee)vF3x|N@^vj|=&LFf*iK7GbqvBB||hCrMbD8_Vza7i8$-lu4iIm0?+fXZJSJH8yy{MSdm_V9kuIp zQQ(SLEJkZZMt<^mieEWh>%J}aa@P)?rubc zh!sJsppMl49V}no%-@O{f^xZh`RcVR=avHa<@48lHRJvRu>-ilaeySgtu4R% zI65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(b dH##vnD=;uRFfip_2wngH002ovPDHLkV1iFdnm_;m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/image-missing.png b/Media/Themes/Umami/Icon/status/image-missing.png new file mode 100644 index 0000000000000000000000000000000000000000..82dd93df92a7645f2775990829e08a7ac7b152bd GIT binary patch literal 808 zcmV+@1K0eCP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00JsWL_t(Y$JLb2YZFlr z$GGXUV{?K@~+y%qupmJ39PDUfubZU6wxd-u_6?7_1vB(}DeQy?KpsnT6!KYjoJ zbRR!~clZlyH?A+IAS?+0+;$uBTQ>m!%5lK!>wz4Jq99DB{pKy~SFhps`^bIYKUUIf z?9Go?f@ifLV8(P!6TSK_hF`uS_3!~wr4qDk7FQ0tp(=$q0)S>+XxS__wqGEsu|xof zFr1+?zL_i#W)Sn(Ck6?FNx;J6@9B*u0fj&VfX$1SBX=Cd0)S4Z^T*fnZ!c>mlQ|t= zSr&4++?hJIENklei-4WRyZXx&O=ib13T*fm=ll3on$gO7>!0|v)Q!66ac^xfbJwN05qjkOew{a zCBE+yGb;d|Qp%qca!)L0TKu2bZ&ilylR`}lnE(I)C3HntbYx+4WjbSWWnpw>05UK# zFfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppnF)=ML mHZ3tXR53C-GB-LgIV&(QIxsNhUI<004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00TKmL_t(Y$IX=AYg=U) z$3M?|e&i%QCrx^`wMn|#(9_nf3sDZf)!?eBj}{ybtg9^E~f+_`?4w ze9+oGquLN3KL%?TmAd7_z;gxoU>||+UdiQReIp~O4E*q10UFFi`}=DHmPIa+P^M{i z`S8{r&+gj2dqiY*GMBUD+BGPb>GV8pYh~q47e4(Bf_+V3z=Z?pbP&FK7ybBgR6bA2 zwsFGYu)w>!otIs@PaPHcvoJAXn57cV+#HxD8>dfmFO}lkt5@$|2X_|k|ECFFhHqYo z$5nW3jZm?OmPnwvE@q*CE5ae$)-`y4FK@9^_uhv}Jy<+>GWao>41O<_0>^QJbUGdU zv9uKYijH!bzF15d z@NruJAF_`$Z6JI2uohG*SW{B~*tU&r+lUC^GiP|XzRuYAxIQHEo)3L{24rE*8y?oQ zl@-G0&LM_@h~PL5(P$JAK?{Yjr>D_x-)6vdwGe!^Cx8#Hq3fxmiG-3!1glU0XaF3? zK`DiZ5IJ>%~kaF;uDGT)KogHr4=aZf;_lCYEI( zB8UiESFUjH@@1}TTJ6_T>DEO^KXn2h-cW{dVCdjMv~;>fyjra?F)=~0SZs(ZrHD*U zLNrRIw^u2{Nc!;ljsPDd1792)88Ntd6MJT+MM&3mdU|^3?(S~vZP(-H&r`i|gVE7Z zGXq~X1N5eKF=U$2{#XnnnZy_zeB41@NY{1d=jU65w=KUr1)Wsh(0jm3%dSCGmL*u}Tz>iKa0000bbVXQnWMOn= zI%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4Wjbwd xWNBu305UK!F)c7QEipG#F)}(bH##vnD=;uRFfip_2wngH002ovPDHLkV1hw(*_{9Y literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/info.png b/Media/Themes/Umami/Icon/status/info.png new file mode 100644 index 0000000000000000000000000000000000000000..037bfb801e11d365b0cd07c9ed2e8a28c1ca2cdc GIT binary patch literal 1352 zcmV-O1-JT%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/messagebox_critical_16x16.png b/Media/Themes/Umami/Icon/status/messagebox_critical_16x16.png new file mode 100644 index 0000000000000000000000000000000000000000..f03bbac9839ad1df1434a46841875a215095f173 GIT binary patch literal 1053 zcmV+&1mgRNP)udZBI$DrMLGIB?y{Aq4K>tWfnYOHN~NqL-^$cn_M_;pDuk z>DzF+0UdXiW@}$kK&C&PhUa5wNg+AHeyzmL*$Y=fL z;Js1+000SaNLh0L01FcU01FcV0GgZ_00071NklBpSNVkW@jUqM<_(6_MqJ5J7l|6c8(MLSpPVa%``> zuk7w{XXA}9&@-B&ncbOl?zwj^@V|%fr-7%ymC1B^KB;MUL`f?6p4)Z$di9{$T-*Rw z7Xklw&WOM>;O510dAXd=7vr7>Q4rud4jf|acDu&**4E>@w*Baz0u=f3bgA_EdZ|=& zcXr?z20Rk+83QRnR#nt32b-H;=kGB_u}+5=ZYEXreyK23bZa#@oeo^nglAdsNyzQ% zFza>X$7E=^+=C?`7xHAjxdlWOHG4a55+QRQ(b-v0Trl4cz+Jfe2XH=aAfu{Re;gi8 z({Y7!0|AnY2$Gl=C|_x22Fk+1Uyo~6Drn8kA(73-sBoDP2L|9#q^|4x3MhYbWre~s zn&cyx4CeYed_ja|+a8gLaykMB4Zt>gy){Qw3+(*|7}Y9#ZheR#4Z1g`Y3TKO`!29O zQp9)kwtLOy^6v4mAx&st(I{9f7638Y)3=yPL61bxI68Vs?C&HHw16~x32^|lCUiX- zPo-|86$Q+;gF=FOK^`B+fW*GEnrkmC`{8py=QR!mxRO+-Lca36Z;5u>Kk>W^f}$kF zWK!(PGHQcC=S`!rx@0>~-U54cpR&M7pg#of2l5M*P@DoTT_Yrbltl$5M&lFkna+L^ z+JruVj=(~Ja2=zc=LnHuYGJGM%JZlufp<3F^STBCp_6dsFT)+3b?NXjKR+XKGXL=l XQUL+rIhAuZ00000NkvXXu0mjf8WHBW literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/messagebox_info.png b/Media/Themes/Umami/Icon/status/messagebox_info.png new file mode 100644 index 0000000000000000000000000000000000000000..037bfb801e11d365b0cd07c9ed2e8a28c1ca2cdc GIT binary patch literal 1352 zcmV-O1-JT%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY3ljhU3ljkVnw%H_00c@&L_t(Y$GwtYY!p=# z#=kpvW_EWv+jhHuD5YAp6$k~X6p{uML8L~lCN)wX(1#{{P!mHM^+6JS@W~j-YA|ZT z%0?kEU>YNe5+6{+5Q?GDvS5I?QlQ=b+5bB`J3BLXt`F=}DP_^CvIMPYAeqkLuPYagk<%xq9lQKeG!i~>SAhP3!N&rj#viu!Y*#GmMr=F_)dMij zoU&!xjzjqJW}nzxC)1&W2TF;F8&5?e;p0^S`UeIVLl9%H?0mD~a>B&b*)-O)`5e_P zG7fHY4kwu}8?o_>>G!(TEh`tRdp_KqHS}V8G!jnC^%HJI3OjqZ?Wjv?9Ip z0N@Y;T@Z*)2{8yEW>zbwRF9-N9Clteg`lrC#F@RkDgXpAk;xZYYZY0y9iE%cmW)Q9 zVo8+fjPtCfTM5%<1+PbzZN`ZJ;xCqrvOwu%RVxDciF9g)FRJ&LZiyDxv<4c2b#5`O zl`my9^IVNfOm?;fIv!cEXro~=x+I{|sp%O5z^UpIFf%-Q^5^YO_q`}yyq+_uKv<|= zX=w?nZ6E*v9EYQ1T5SB@pgPrYU+9TFdp>p?+xm1q0G(SnH*#wF!Q~wx3ck9lbETNk z%H#f8mogep@3w8{r-EVauT`YAnqePTBzMp0;Zx?Bk&(mu_U`_q$`t`{>^DY!AHI^F zp71b-*m{{wsvePmV6tA3EdViau_=>Gm`9F$Kjkp{HQYw?0-}+y#hKmr?a+`E@Om1@ z{`fPVPtP_`f!No^ksTd(BLq(JsJrccLk_G;(Do+n4G*`L!m!@X86Zf zwd~oi&YnH?ky0bRRnRg+qi?og_;*`xRQ-pid?#U40^%5IgtQN%>WYBb>ESTW*(f5Wo zJY0s?72_K=^kCDbXRu<$D){RuT3g7SyaCK}rgPQQaJa%cQM9N07?LK0L2QdDnL!;DF7M(1pE579nEC2 zmwmqarH2lEy#qiJz$}1F1&{_{R|^1c9)JwMQ~9OJV-+K-GQBaM@&Bg30Z*ghy0+i* zp8x;=C3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK z03~!qSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<udZBI$DrMLGIB?y{Aq4K>tWfnYOHN~NqL-^$cn_M_;pDuk z>DzF+0UdXiW@}$kK&C&PhUa5wNg+AHeyzmL*$Y=fL z;Js1+000SaNLh0L01FcU01FcV0GgZ_0007-Nkl(32k+30I>g2IJ9#H)EniuLdLN&4VV0pTrniNeK7>NpvpRZ=3EFiHCTl#f$TiLCD_>Jb~=RV$;c}?n>32|N`Ur`^PT(lAG|C!?FmRZ83+BW z;;uni!H%-gRXM(9Ai`hk+KDJj>ciOZND*Kuksx)|CHjJqr6d+zl+4g%nuai8Mew)a zu+B@l^||^@4Z79Jbd^V4Rs(`lHH}oXby><$aoxZf3jyLeMT;?r)Eq4G(G>I&0JBvA zd4Ri9tIM6UNg;<*=bNL!*?)0{_(b*!0(_FE^@2@TC8dO48y@Uf_3oz2Y{8;zv~-yntBe zFs4q8jSL$@p>WsRH~oqKeq-K{^;$8IsrS~MRM#^OLs@8DURv0S0o^q4^!Z@#z1uyN zeN#%I=!nJfaA@e=4-1xq9yiHS!rT=LjaJxV%XWWPZpp9r;7s|YX^BljX+B_zmZue^aYbl%8A+8v^4i~v_{u5 zKvC51L!pRZ`Xp0XQ+g3bR8T2%Z!^UVNe*j~X2|st>@OS_K002ovPDHLkV1js@6*B+; literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/messagebox_warning.png b/Media/Themes/Umami/Icon/status/messagebox_warning.png new file mode 100644 index 0000000000000000000000000000000000000000..f77bca6c2e8afe2ee1891bc545994e422c6407ef GIT binary patch literal 1079 zcmV-71jze|P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00TKmL_t(Y$IX=AYg=U) z$3M?|e&i%QCrx^`wMn|#(9_nf3sDZf)!?eBj}{ybtg9^E~f+_`?4w ze9+oGquLN3KL%?TmAd7_z;gxoU>||+UdiQReIp~O4E*q10UFFi`}=DHmPIa+P^M{i z`S8{r&+gj2dqiY*GMBUD+BGPb>GV8pYh~q47e4(Bf_+V3z=Z?pbP&FK7ybBgR6bA2 zwsFGYu)w>!otIs@PaPHcvoJAXn57cV+#HxD8>dfmFO}lkt5@$|2X_|k|ECFFhHqYo z$5nW3jZm?OmPnwvE@q*CE5ae$)-`y4FK@9^_uhv}Jy<+>GWao>41O<_0>^QJbUGdU zv9uKYijH!bzF15d z@NruJAF_`$Z6JI2uohG*SW{B~*tU&r+lUC^GiP|XzRuYAxIQHEo)3L{24rE*8y?oQ zl@-G0&LM_@h~PL5(P$JAK?{Yjr>D_x-)6vdwGe!^Cx8#Hq3fxmiG-3!1glU0XaF3? zK`DiZ5IJ>%~kaF;uDGT)KogHr4=aZf;_lCYEI( zB8UiESFUjH@@1}TTJ6_T>DEO^KXn2h-cW{dVCdjMv~;>fyjra?F)=~0SZs(ZrHD*U zLNrRIw^u2{Nc!;ljsPDd1792)88Ntd6MJT+MM&3mdU|^3?(S~vZP(-H&r`i|gVE7Z zGXq~X1N5eKF=U$2{#XnnnZy_zeB41@NY{1d=jU65w=KUr1)Wsh(0jm3%dSCGmL*u}Tz>iKa0000bbVXQnWMOn= zI%9HWVRU5xGB7eQEigGPFg8>%I65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4Wjbwd xWNBu305UK!F)c7QEipG#F)}(bH##vnD=;uRFfip_2wngH002ovPDHLkV1hw(*_{9Y literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/messagebox_warning_16x16.png b/Media/Themes/Umami/Icon/status/messagebox_warning_16x16.png new file mode 100644 index 0000000000000000000000000000000000000000..a50036cc314f03a48af1b3aaaeacdde233eb8c40 GIT binary patch literal 1047 zcmV+y1nB#TP)udZBI$DrMLGIB?y{Aq4K>tWfnYOHN~NqL-^$cn_M_;pDuk z>DzF+0UdXiW@}$kK&C&PhUa5wNg+AHeyzmL*$Y=fL z;Js1+000SaNLh0L01FcU01FcV0GgZ_0006`Nkl^Ki0`&N>WlNIPiGPzM1cQW|#&3)7Vk@=K;?p&L42b{(P;~o<0Zaonob< z1cu!1*=!<_w<;C<+T8p?aW6W>@=hOz04=8L_u}+8Aj>$dX^UM%bZmhUNxEgI>RFM^ z!aq8SL?Dp8!uic(2`mDsw4&TT6AYp{HHGT-wFMLlkWSNlD5NXqw+g*gO`4PYC$EUby0Ms1p;&1OmP1IKSIUaL~d$Fx(#sJxQff zLd?y9sVY=THVgw@(I~jCLu_ooDwT|wX}@_7`a%l;# zmzSYX8wv@Du`vurBH^GA3rz}#;2Mz2>iVNZJnj}VGhlUEI?KW-n#Euu0o~_=T`oh4 zM1YkQn3k24cXvO20zNyd{Ltn4eKC_!qkDU>o6@!oMNu5DLfhCd#yKQ9-~E2PU0W+X zuGKVWm5a9htgyX(!;)oDWvtN_BO%62(<#81bJ_G?*d$Q@RxFy09OtcB@G9W#rJ1|~ zd_RCd8BnW$Xv~R%DsI5^0i`tXbD#WI004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3labT3lag+-G2N400SyXL_t(Y$GwzKXcKu9 z$3K&qsY$F#O;j+n?ov`}yIZxj!WIz^Dhi7iQSlF6t#}Y&7xu7hQM`$u_!m5Q>8ceG z@uJAyMV5+Gjjl~h(zNNcNw8BjY0b>R9w`Yto``@Q!*^Ly|6e(;S=Dd%wd zOwV0lPZc?@{q^UmV}%c`f`C0cw(Z34Zz}7UN+r2{=Vmuh6#&1#iCu@zmi2pl`;U^f ziU2h=H2`$|(h3qJpeXq#5{bf-C$Hv=tq5S-HUJKX3Q8_*Aw58NXnfw-iU2|g08~xQ zuSkrHNL^#}(3^~YiJITgkh-?;H2!s#_+s?gbvrI=K{LtCyR$+fpiF92DU_3f9uYx@07%q3hwlR|?q5Jn)+tR|+yLZTZ@)>}l;B0Ot8VGz6&{tHVcsve()9ECYN)`QiU0q$S z2S<LL>eAd*RhqM&-cxLaB<;&J)GFb09k6sG9(nI1`nrcRiw?!ATA zvNOKMMwcrOPftuR5{U>inH*FsYkOhP3fG`_R!=;@QlKz#7BCH%z(+tX0qr~-birN$ zKLObAHUoW+;oQAk%=fYX0YZX`4@34%o%Ze?;OGw)ac001R) zMObuXVRU6WV{&C-bY%cCFfuSLFgYzSHdHYO`e7ytkO literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/network-idle.png b/Media/Themes/Umami/Icon/status/network-idle.png new file mode 100644 index 0000000000000000000000000000000000000000..501d3f6fc92c962e5aa75e371f3144b4cfef6dd4 GIT binary patch literal 901 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wiUjzCxH6Ch>g($m7#MPMb0E5dgM%RqGcz*?qp77Hg6iw*>gsAM zD=W&%O6%+EYHMrEJSyDcy4~V>Jd^r8lO}j3_xq$z_DP@On?5-(Z9;J7(733eadA=evZ9uyMJ>xp+gFsfuPpCcU0+|<*x1n2Qa@qBgt>F) zELpO6?b(!re{ISdR8j7i?^F3}}>rkXG?FnsWI zaSYLz*gENE&>;r_*JdXM*8oP*$h&TxjNvSt7cwXKin%%&{Q2J=U^;W6N^tA*uQp$H zhU$9$pLm?ta`}OUCx1FlH{d*8)GfMl@7`-#SG&R^p6p$B)%5K_?P*JH=GfTYyEid@ zeTMCQ?XDmdZxO5bkgYL~4qf$J?IU8na{huFnUh7Wc^?-W%vgNGa`6KH7byh`SE*i` zd2&f2$L_kaImwG;cn(S2c(Q)KtnidS%NHCr6tz75Nr$1vXFqpNyJ?ifn$34F|33ER zme^)q4u-jP-w%oT_3@ospQk4r(&9AnN2FHiB%3?j@7c?qWcYl`t)6ZAKrNEng)jRo z0|SFyiEBiObAE1aYF-J0b5UwyNotBhd1gt5LP$ocg1e`002h}=vZjKOfq}7tS7LT) zl7f0xW^Qg`vbsWWNor@#v7>o>z40H`Fbq&ly3@xmT zO|1;gv<(cb3=F1ywlGA|kei>9nO4bQWMF8hYhb2pXdYr{WMyP-WoQY}@HCDi4wPCw MUHx3vIVCg!0H)njU;qFB literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/network-offline.png b/Media/Themes/Umami/Icon/status/network-offline.png new file mode 100644 index 0000000000000000000000000000000000000000..9d2c7b7524e592ca36eb8997b1c215d1ec6a1c69 GIT binary patch literal 753 zcma))T}V>_0ELg0VT!fNpUkj}kTYa>@2sUaOJ`1ZbBfLpYBk;F)H$YW-lYZUoL8(B z%v$s>K~u@JAMqu%2OnCE$_GP?nLUJsmbwj=FJTy~r`|g7!Fl=4)45b!lph(sI~+hH zBh8fq*cBvfBY+PBlWzcQsudTCfnj^Znkp0kp^?f9f%8d#g#&&9uq*;*68K&P(6@kC zjW_n%tV08A9i6cYSjtXx|!(1T#Ha&01MOP=y_p7IK#wlnH( zMq`)NI%Kui3u_%EHQl8;$7$0YRdbKNrK{26Y;1QLEly+m17o|>)X{I^`b=EEx$B|1 zYoOUSXyw|j+6rRS0e%JVC6-n*bo&R6T2%hg-XxJywKWlbLjRN^$7qW>Vo{-tfLyz z4B#b`D^?_5?X-KgT1g(Feer{dE%TkvVyAUlJX< zXUoFo{bC+pb~&~pHCErGt&v<5oz&~g=ZC_&4ZePda_^B5byj@+9Z`0Ow5RFx-ndh} zv@SYi)Ny{>m!iA^?~h-O`8l!og`7w=Cy>rb?nykuKO1*Tu)v#Wh;rF&Zx^=OZOPjo z8P7$G4}Y%gTu5*43PY kkfbO9DH2eJ6qJw_9-^s?zrt5#m@@b@MpBgfB&W*y8^gs9H~;_u literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/network-receive.png b/Media/Themes/Umami/Icon/status/network-receive.png new file mode 100644 index 0000000000000000000000000000000000000000..d0cdddeb2593b437c24a24c426a1e61abdad5133 GIT binary patch literal 1027 zcmbW0e^8SJ9LB#Jpn!Uo5keupX9{_=dN*>w+>*gyn*t3syk!TujR6ba?d5epz}()A zcA`?k2_lE!b^O%8t|DfQegFqJ(hTCqZj?1WyC@hSLxUaXb+o_wx96Vc?)m5Q{P&y_ zW%0L#z7Prk+k^sM4gf@KBsc&7oY#N&9smN1xS3pl&#Jeq%Ypy^Q3!G}0ZzpLn5_Ul z05H!2=um*ajsvh70CwU{_G~r)gy4d=P6BM%u@TBo_e|>m6s0Ibk_P`vBxwQw3_Ibc z#bU9)Fquq$sj!lMlcdRHGM1N@>Ge92G#QPjnSwHQUKKI9yEEaQxm8_|$X>M+6Yiqr6qr0c4$7Zt)4-Y#Ww;T?~`1tt5#Kh#} z2$eVGc&H)+1a_dxw%Jw&(F`#FRyw$9*=i*VPRomWp#0JacOC3`H9c# z^@=)6UI%~}ZBCXLr~!eSL$*H?_I&uxh!>-G?|EroTpT?yF^PRtns2<^{awFfX3h5} zvfKm!6!{T856%Yd5&%$k33*&eL}bvwH|poO5zz1I`_b?;jZI$-F#y%oS5i1$+6KXWLbHeRJXe9WP^=WRjDw>Z@PZlkzT-LUmg zGC1};?(GSp_7@Kp{?aZ%VC;k`H@n1w_fn&OD{p?%=-Y0cT8VN$j3q4Y^-b=Hr<|{o z$KHE;M1A&=h4y7?>bSnD!SI^&)gmKV`XMqmg&7Zj+}$b2P3}oLnwJ{|f$ZY0jt5<* zIEPGX{=iVQh#hk*a^QYI|L#ty&m-p(Uu|-$hYNl^y_O|?r~Y0G{z|xWe->s44Sds1 z;iTOi`QzfVQGg%(w(-DPyD$&uK1RTw5y#3Ev?Ql(ZB zvGI#i0NA&&R3OtTS&DyWfPpel8k$5ynG$*;EB*kB&WuA*7K+w;6Y2i}FinA~upAgD kolZlUGE2?X<58aQjS;r15lunDgXcg literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/network-transmit-receive.png b/Media/Themes/Umami/Icon/status/network-transmit-receive.png new file mode 100644 index 0000000000000000000000000000000000000000..fa4c15a57a2e6d0dbde05e314703709f2e9e051b GIT binary patch literal 906 zcmeAS@N?(olHy`uVBq!ia0y~yV2}V|4rT@hhU+WOo?>8NU`coMb!1@J*w6hZk(Ggg zK_S^A$d`ekN{xY`p@o6r7Xt%B!wUw6QUeBtR|yOZRx=nF#0%!^3bbKhVBjq9h%9Dc z;1&X5#!GkW{xC2wiUjzCxH6Ch>g($m7#MPMb0E5dgM%RqGcz*?qp77Hg6iw*>gsAM zD=W&%O6%+EYHMq3VwX7PZgR}s;*`9?rEr@|;dYmzZEj_|+{$*lm+kT{-R@hlE4J-W zV)v2kiuU{|C-bMCESPe#VCu=D>1T>&oGzMirgZkX@_85P>+2dD8=6|`Crp?yckY}e zOBSzPyJqv|&D*wZJ9_l!i4(_9oH%ju;>F9CFJHNGTOWy7}V8 zi`TDTzyA3B&6_uGKK*$6`Nz92KR$f;@afyH&!0bs{_ow%z`&qZ666=mz{JeL&dDzz zA}S^>AuT7bq^zo~ZD?U(X%iL^S37;p{sSj&{rvS(;nN%j1_s6?Z+91l4pvzY1_p+& zo-U3d8WUS5z4SZeAmY~Ta5E$(aLa-PO3Au<8dtW(t=pu`ap}~rfA1ZQE<4@!{PM?k z=AF-~${ha-8>KhpHOiiT$ZmT`OTaDhnJD+7u=uN8QHpkk%K5X_z5KN}-lro`;=KQM zSEH}p-#_iuIbEL86!W}$kcGC>nYF6A!}`_QT%ak}Ti{uLc=+xh)3zr6A>&Sn}%e&=pa!w=WDM=~zo z5wYdB>xZ8V3=DQ9t`Q~9`MJ5Nc_j?aMX8A;sVNHOnI#ztAsML(?w-B@TwEH-nhHh+ z2F40riP@z|3hG&zxw(nS>I%UnshN4HMe4@yBN!MMR6!=CB$lLFC4-G&FfuSQ&^55s zH82Y?w6HQZwK6o*HZZURVeP?Gg@^>bP0l+XkKW;9}G literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/network-transmit.png b/Media/Themes/Umami/Icon/status/network-transmit.png new file mode 100644 index 0000000000000000000000000000000000000000..bf2513307db0b25af9bf5c4175d1afa7ebe74246 GIT binary patch literal 1030 zcmbW0Z&1?(9LB%nPi`1x3W{*m4-IB;T)zz&Y@sqXU=v1@?Tj)eWBV~Dj?JIjfF=8F zxDylZSOLNb=;g>D#5dQ=77x%cGd;B^Vw`WW0 zvL*G|dPlZ?AV=@umo)H8tbAERuI_rCqA^d=Bv3R8Djf$(ts-S(5q`et^bKi6hqUfy zS!J!-WK)}M>L!=E*`+brH0DbhvrW_F(lop9QuS4nwcWb5 zA1coFR9v`A5Jt1vRAV8mR%=H`duOMuxA#haf4|*s9~l{OI!B#O=lJ-z+wGp1n3$ZL zoSK^Qcs$e7(=#*Ev$L~vb8~Z#9?#Fu&#$a4EG#T|*A^ES7gyJomX?;6mskGsdA(lY z=ViMAAYNOLFM^=pkPVcVH$_Bl-4+$|+Uq-Z#w8}w($bje2MZfMkiKQ;|B;*?Uz94s144&N;HxRsOIV(cIdcU^&0O(@xI@-Fy8VSKW577RFNEaxcNRi z^%6DLwc5I+=51&f{P?@hz{TQy_D^o5H5qucp6B)cKm1J^?K%A*e=;QlV*XNP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00bXNL_t(Y$K_UEOj~sn zo|_t@4;p<_E2*oS|!1qKTxaNF)!1-0K{Y*gG2orjjFm2|IV7jEEh@7FU;MoKqip2pU%vLi%2-qTlo4UguGm^E(a0^ z4j|^RDe`n*?h@~1(+2W=DneE?+lpFzMogT*Bu?IjU>}<(s|*Z@QrHTSso*H@{}R|K2fro_0ELg0VT!fNpUkj}kTYa>@2sUaOJ`1ZbBfLpYBk;F)H$YW-lYZUoL8(B z%v$s>K~u@JAMqu%2OnCE$_GP?nLUJsmbwj=FJTy~r`|g7!Fl=4)45b!lph(sI~+hH zBh8fq*cBvfBY+PBlWzcQsudTCfnj^Znkp0kp^?f9f%8d#g#&&9uq*;*68K&P(6@kC zjW_n%tV08A9i6cYSjtXx|!(1T#Ha&01MOP=y_p7IK#wlnH( zMq`)NI%Kui3u_%EHQl8;$7$0YRdbKNrK{26Y;1QLEly+m17o|>)X{I^`b=EEx$B|1 zYoOUSXyw|j+6rRS0e%JVC6-n*bo&R6T2%hg-XxJywKWlbLjRN^$7qW>Vo{-tfLyz z4B#b`D^?_5?X-KgT1g(Feer{dE%TkvVyAUlJX< zXUoFo{bC+pb~&~pHCErGt&v<5oz&~g=ZC_&4ZePda_^B5byj@+9Z`0Ow5RFx-ndh} zv@SYi)Ny{>m!iA^?~h-O`8l!og`7w=Cy>rb?nykuKO1*Tu)v#Wh;rF&Zx^=OZOPjo z8P7$G4}Y%gTu5*43PY kkfbO9DH2eJ6qJw_9-^s?zrt5#m@@b@MpBgfB&W*y8^gs9H~;_u literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/printer-error.png b/Media/Themes/Umami/Icon/status/printer-error.png new file mode 100644 index 0000000000000000000000000000000000000000..6909011b361e34cd26e53dadac112b6c7b8a31d9 GIT binary patch literal 1169 zcmV;C1aA9@P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_dialog-info.png b/Media/Themes/Umami/Icon/status/stock_dialog-info.png new file mode 100644 index 0000000000000000000000000000000000000000..037bfb801e11d365b0cd07c9ed2e8a28c1ca2cdc GIT binary patch literal 1352 zcmV-O1-JT%P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_open.png b/Media/Themes/Umami/Icon/status/stock_open.png new file mode 100644 index 0000000000000000000000000000000000000000..def61fe71f94ef2eb82d10d3159a4ff1e5e69707 GIT binary patch literal 1073 zcmV-11kU@3P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY4c7nw4c7reD4Tcy00Rz5L_t(Y$JLd+Yh+av zz<=k?o0%kI+!+=dqlF*;!a^Dmto#E^If7szmMK!Cnnp++5F#k1vaz;z16hdJ$P!(% z;D$AuRr9sUWMyam`*2UT^?%F4?F+V>)-Dvul=Iss_(dG`JKN9kAuWx%)h>r|?BM8GUxU=$3y=$>L z0(hSnxd8Ms`0V3%s5pm}`xyfPx!5Tjki6-q1mUbVr1RC?<`pR_wI}(-U-!G$n%F|> zRY&jvC+=_z;JqJBag2%+p*si+vv_UeQH{Y+=$`ix*uNZsh7d3_o~6R_&d+|jxf?7@>h8;@V6ajV8p51&w}$k^*p zp5^k;6Fazu*7v?)`%gLSrU5Jv!s`+PzbVn3?(Ghxa~zwDHA0 z`Dj>?!7A`7@XF=1{Yz2c4A=!ycs`&4R4$CmO)QtPrGRI`)BB&x-yN?rIq#q_-~a#s zC3HntbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!q zSaf7zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf1@qhs literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_trash_full.png b/Media/Themes/Umami/Icon/status/stock_trash_full.png new file mode 100644 index 0000000000000000000000000000000000000000..a6b9cd3b5498f66c6a0caa0481b70399a2036e91 GIT binary patch literal 1589 zcmV-52Fm$~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00k0BL_t(Y$F-GdY*bYg z$A52~c{9_!6d@I{g$hLxLlh7dQ9(e7F$Oo{7h_B`Bx>SEzxmz7#2Dj(i62l>O$Za9CUbyJgL~CtNo}vG-^+HdZ@J?&^*l<$iDECGeZH4SeT6|i zrr{|kGx=H&R9@WQ6UB*Ndj`Ps>RxSa=vw*s&gW@wZzq?{a^L+AaP*f&EM2mc-r})z zpJ`dxwnGqx%$wK7!UfB5-IZJ%97^8Jk<#g1A)o&W(e7lwHGTV5!!X{kjFtBKb+^*h z*$G;cah%j)cXu~!^XJpt)WWjv>ycU$g&|T(8XKFbl*`p@Zz(_wf~}q_cAgYu2vCvTd?n zj=S&K!pRfIak9M)vj9@c+`U`YGrz4nrEeJqdM5u`r&2JqzRaelZJ&I4h+^+CmUXWn ziXwF4)UgPoTHP!_9MyKc@%EE@q!d@I^NhKkdp2*QTJ;Hn8sC2XC6#KQ4&}rPm&x?0 z=klXlw{B?3xEX}h$Ycsd6iR|n5=AOWAhpSuNnbEo69hGAZrH+Yw{PUf9}aWw+#i(7 zr^z?|)W7@nv89*E6pZmWqQ&zp98=;r3ARZlYvKtT5i7KIG00L2e1cky(Gi2TmJSYo z_YFhC-;%43$v2EKUI_ZH&e-OH9K?YB_DwX$rK`Oce#%&@9MiC;$G-CYPD8r>9 zQQ}xcU^q0^FJ!oMhK7Q|pBzW3I2A;q`ieh+RR6|6F*07hq|=0}7^x&-JkD_G7)?z} zsa8sa!8sPUufs50yzD$gEEpK|`NvN`Stl*F0MxG7`_%A~k^&~Cpa1{(6MBoS^_w}4&Hw-aC3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9bV6J literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_volume-0.png b/Media/Themes/Umami/Icon/status/stock_volume-0.png new file mode 100644 index 0000000000000000000000000000000000000000..35ff977672b2dde8788a5b8ce67ce8ddd8a05dcf GIT binary patch literal 888 zcmV-;1Bd*HP)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00LD>L_t(Y$L*6rOH@%5 z$A5R;8|RHO3Yk)llYIhP^#xkDQ`3|dLM{t24I;1+T4WGmEi`IV)Hb zDHB3LA`8c6oO|zkw?%KpQ7PwHix&OiaL;}B9)9=y4-fw95Lu;Oygzhr@MPVIsWXj@ zTMG+|v!&LBO1YFwCd-wS*Lr*UVyorQUb;=FdsbOwgs{XU~TA!c7TdOQ9H#)#N`V3Cg8)1u&I=*l1O{ZX8EiX@)|y;SA_Qoy zFo0IO!xlzN&(jq{b${0a1WlE{9SQtu+%9sqdb*J6tJce#Xo`?wBW;^hyHw zkR5>Gkzx12!yy-7eSPiacxpUT9<9J&P<|&tw8x^)8((7SblQ?qhP2k9rKP2vme!`t z(dW;)8csJjN-7A5C% zI65;mIx#jYFfckWFqZQ;F#rGnC3HntbYx+4WjbwdWNBu305UK!F)c7QEipG#F)}(b zH##vnD=;uRFfip_2wngH02y>eSaefwW^{L9a%BK_cXuvnZfkR6VQ^(GZ*pgw?mQX* O0000004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY07w7;07w8v$!k6U00U-8L_t(Y$L*C(Ok7nI z$A9O&Hx85lmQrG+SQqXZTU#J*YGQP$F0_SWCtZ+&+N7GQw2_*Sm*_%G3_%wK(Zou} z2vZtOiW`lx>W56Qm>5jemP8?GR5TrAnEO7Ci}_#*L!@1~@FX|y+_eG z{w3`77WVeN{pOat?>f74$Br9QQ{PRjcb&DKBG+{{c*eZk_0qA*K!evy)ighQ`q9Q` z1|qiwn3?&h1Hs<9wSgcA%SzQ~$RSHBdTtpIJNWdgk?0+7*!CK?>ADtxvBRUPbnNbk zHx^Gj#huo+eMUqCAPhr|Mxz5&Rn_fBUwFR!+~;E@_iz05zFz4VI+yG}_(fA^{6$Oj z!|g32jo&9)%_;?|8gYAE$L(<)AW~OX*M8!a?uwe4+O;KGdJ2)(_8g9HOb*6A0rjHQ zOM7lo(6WDTHEl;LO0;ot*l%|p% z0L;$)Qpm5(Tm``MOz!IyKby-c@(4gcFfCX|00`}G)e5@T1f&2a8ITwr{R=1u z$Ye5jp27Eh&YVg79fr&18e{SvGcg#ujO&IAbF)Vjot+qreR==k_(oJ8azxIrQji!P zT>?1Se=>OI-M#=|Zf^F|k;KU2hVlv&#q+)V{>u@{HN-jr5cRbWk@%&7*pDSW`0e^1 zm8n$9HpW=K?_1N;(@QP;_s$QU{-|@?z1zw>;}xE1K9cXpJ_Bez^zjplJ{YNpJWxDe z3~^iu=lW6v4gfEg-xgryc=23b;eRjw0j92hTU?Le&;S4cC3HntbYx+4WjbSWWnpw> z05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7zbY(hiZ)9m^c>ppn zF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzC bV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9w-8h literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_volume-med.png b/Media/Themes/Umami/Icon/status/stock_volume-med.png new file mode 100644 index 0000000000000000000000000000000000000000..1a559245ec2be43dd6bf0a9092aabfca49d20c6e GIT binary patch literal 1008 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 bV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9w-8h literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_weather-cloudy.png b/Media/Themes/Umami/Icon/status/stock_weather-cloudy.png new file mode 100644 index 0000000000000000000000000000000000000000..2b13588588584e47cc3ea9e2346e986734972d84 GIT binary patch literal 1078 zcmZXNc}&{{9L2xQ*u;`KQDYaB*iBh9gZXQjFe(~h5GWgntP>Gk%4i3;a5#!?`cE@0 z^c+XIGLDfD${8YODT8sW94+No&oSshFOH73AVkU-n6kf&$>$|6-|u^QtoXR7<7gi= z030VqMS_%LjX<@Np03C&1dnrc&0Gk?}5DW0z836j90J{J{UjfKN z0M^I=gh~KRPIdn`p#boDPIOW(fY;k@q2PY?NdS~6{xT^N0GMnBola-68UG`n$!4(G z3>L?2m}~}z$7FHx+1z{%kIB8q;_#Rp9+S;2I5aFSkHz7-FRx-8ZUKvv&pQ0_3fXLK zK}$Fs_rG|4-Yg|#36QxqFR4G(yg-WedYm^#|O7m1b|Lp0rh53br`NhSBMeTxCtJUfjb$YE% zuhZ*wdV|hj&>NQYh9!g1xMVafnM_8r*<`ku%oel7VzF8+%gff~mE{#TE32z3t81%k zZq{tJRhw1c-pl>{ z{r!?WzBd4%u3w6ZC@J@FU;$7kh!J6=k7rZKbsvyTiQih&9>0@z72h(czCwxgrdav! zVi0?~$g- z^oMN?4cKr`JN_uCLV{22Ya2xTzEmdjz2>R0Usb5e1RgGz>kj+$L;s*aTxjSi@rjR4 zy-`L^MD@!)Mf`^+F#hf6K=gCG$D7O71ve;1Ai5yQJYIQ!5CD8>k>AlSW@lznvuJSf z7Bz)NrC@)}prvC;=~QeKF*e>o@yh}LKX;IlLZcE=4^x0g@JIj>5`YAea2E)HpAm3D zXAy*eAmx8w!2PF?nN7*KaR=}S4i|s~1>k~7I6MI#Ou&V_Dp;<0UIPFiM#e?7hm-IA E3neQvFaQ7m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_weather-few-clouds.png b/Media/Themes/Umami/Icon/status/stock_weather-few-clouds.png new file mode 100644 index 0000000000000000000000000000000000000000..bd1dfc155d97197bbeb4e372190f1991aadf3821 GIT binary patch literal 1277 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv0RI600RN!9r;`8x010qNS#tmY3ljhU3ljkVnw%H_00Y)ZL_t(Y$K{htXq;6P z$A9;J^U7lqn@rQeJX-7nr4Q8lNF#oehFY~?Lkgv$A{5e<3T~8w=%P?8t%6W@;=+{+ zkx4?0f)wmnG^v)ck6`Vz64ItkOfv0cl9_L2?!!e&P3wdtJHfNM=iYPv|NB1=_+Odo zaSy3p_tWL%$Yn7|x`EeJ++A|szXDa)R64sTm>fn7i)cs6`uv?( zAt6AhZ(n1XhiJ9s^qfS%JCw z9pdIC@azR&=cz4AYVuv?XzU7+$04}#u2>=bb4N3VEes-B)?!1o7GqB?gmX`c?u_4d z9MNsj`((HY%sAi&o1gnR3ap;ECo!7w@? za01xkC4>xOtpal!%na7%<_OFrRMlYxj(9NiI%CDXs@I*x@;R0-d|ipl|0uAek>t^~ z8fA?wOgh*+=uxck0|rbMOqz+wEN6#mNe#EKCYt3@IYG1INgND{3IugnYRfiR#obmU z-8K~2FQ0xxS~-j`w*iB-d2+KEzU*H^du#{Q;Tp!qZg6JsG&@)P`2FhO#$xIE9__dO&BSBWw6nLt257B5Nx!f$dT#g%~*EyR!d(9YAFQq)8 zwSMQ|yYG!Gyaxcn#Kia@CQu~t$4b_$d!Dvu8%dr^@<44JrDBm{p@8S*5ssu|`;O-3 zre@Bc|7~ybToPd4f@7-Uu8SidVpA2_D?_CIgyyCe0)YT+t!;$EVU$v6ZK$t{uxaBK z)~#Jnu~a0|&>*a}dyX7E+_LZvXW$W_J^B~{IzrEi#U#h-yc9JO{rAE^E|vl zo~r6<3Z6$gJw+yyVJ4ep{kjc~<2c945~#Sl6>YbX)8C?OER}7jXliU?W_E`0iE*aV zQ%q;4@d_TE=P^2Zjm&h0n`1Xf-kkIc4ET)EF9QJ< zF)4%+Qhu$qKDb!yJ$B^iVX1oEA-fdBLMHxvN8Ez+cq!vFvP9dt!lbVF}# zZDnqB07G(RVRU6=Aa`kWXdqN*WgtgMO;GP-C2jx!03~!qSaf7zbY(hYa%Ew3WdJfT zGB7PLIV~_YR53U@Gc`IfHY+eNIxsMn^Efd8001R)MObuXVRU6WZEs|0W_bWIFflPL nFg7hQH&ih)Ix;spF*z$RFgh?W9L2wEx`BkLGpN}WqCzwb{Iy2V&TIq9pfCtW|JX9-Kw7{|4zYmKjX4E{ zUR*|;Vj{^{@Sva!bjz|pX-ivrOh&;%4|@E1P}<+Gv`{EzES0~OeBK}L<-OOOlbsf^ zN=K;X}ph8~I594?N}#JpOIy>Co`tgiI=vzd|9ED<kBv)&Vu?^XF)p1DOJ(B|aXqV zCY`~oGn(}#^SlY2H=za-YBZw;Gio$jjHuOQv6(D3v&Dv5t*F&*w%Spf-C|p?*c}$T z!)kZf78Y#|r`_RPSadEpT@I&f(dlx!utf}WVYmy!G0g44-5Bo1@g)rRU~V_=Uc%f< zxZ8s-dGIB#+vD|kmp$HPug|yQTV3(3`c{LiuCA@F1zB73ula-c{pH7q`XK-uW#nb&!)wvi7iwzjE?#T6&SZ{^@&!WCxJ2q+@_4*nuXlNQWyKe~ z_WJ{Yjg5b&cKuWW03l~G)5x8cig)_~Lf)W|lcjlJ}IDbB1G` z7*CZ?D<$vVms|dAJX=C3zt^Rs8l2X^>uJ!=H`;62t7%7$I<@5>$ zq*T+37<5`x-C0Ii6t#>Vl}5?R!D!JH01zE4(ux>#Qt_)65D_90hn$Q<5~zd|q+`cP zgoIcGAtA^u&k4f+2IZBsvnAgF5g`!bkc2owB9%ZS5fe#-lUoDybm(aS07_~$nU`F6 F^*@l(TxI|O literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_weather-night-few-clouds.png b/Media/Themes/Umami/Icon/status/stock_weather-night-few-clouds.png new file mode 100644 index 0000000000000000000000000000000000000000..bb3f5c1429288daab75290c317bc7afb57168da4 GIT binary patch literal 1269 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 V1|9}E22KB4zg>HMs&`P5{sQ;N?J58O literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_weather-snow.png b/Media/Themes/Umami/Icon/status/stock_weather-snow.png new file mode 100644 index 0000000000000000000000000000000000000000..18e9843178a3e139189aa5a96927332cfc4318f9 GIT binary patch literal 1507 zcmY+83rv#-5XTRlo5+Z6zKGKd9XcmL+cH7R96@1#6NAF05@jNw2;(I%r}4$KwBNIZ z2|7VkWD_jn3qe6jTM8nyls;&M(o*^WERTYM%RpplS(bSD-R16b_rF6zcLqAoTQ&~> zoP&e>cLRWn&F9Vr0ClOP82~;w;uqovaG&k;Hfjz4xJC!<4gom12H@NUfC&IN*AI|E z0k9kdpkD@9no>}`>l*-YObObT2H>>N<_>VBzy)AdY3Ppd005Ytp3ce1AxRR$Fc!*q z57vb&G;5}VqFD%*h2TsC%R+Gm6JfBk33fKl&L&8XEpl>*na)hf&c-pEg%NiCPi9Jc zB-t2>Gf_MXBUlK|Zt15VPwE_X}d2>^9 za}x_?5F|pfQEpkuy$9vByr=c`0=c}6g)&e)3&olC>j+LDxcxN2#z+pv$ssQ0W#`(+ z$-R`5d+GPQ%NPI1&%IJ`IsfYATP4M}xh34P5^j0vox9xfyWEEr_a8pGUs?62s;Z*u zN#&F3%IccOH8qcGYioG?r*-_=IzEpt;0XkE0wGTztP=?NLSday$QRZNgd%>uNLb$> z5Q&7M27#zS*w9el_)OH;)X>!2_`Lbq^OmL-akE&`B54)3wo0T@iL|X%Dr;@CA#IaO zWj5Mm3YlCXQz+yLrA(oeE87+AHk9ouMY~F=QY%&JcD1@){X+GkL)~HHg+|k<>FVs# zc51a+S3B>+kQgSYBENECU0Ufx&^n!GXb{!J(nSVe7DU#5!VQWOQ_7^wsF= z*RRIM$0dOxD*zly-@P*o96oY%TDE+}%2jk9A79^q_@vad6Q@s~J)3!fg<|Z&YehFp zZ{NB5uu3YE%a!VmPEGgBa|WY%XxM7Cj*Pr|JvKHz{^rd;6B84YlW*Tn&DgvDtib*F zCjj7Z==(r_hv``!Isk{lV1K{xwbJN=uAkrI`qadwT*qj%2b;eu)P2?~YHVV;iQVF& z{#=pzxBQ03jp`*oZWX6EI@Gxb9+DoYITaUhWGeRyoiN4mCRfwqxlUEkS#z|JNMJ7A zS1;Q4vubIwuW!}CcQT(N8l{jf0;B1 zh%oND%16D${pY4$W(J$+zh^GIm+X!{sG58qd{xy6_M)!t0!^*#4R&OsKB{Yhed zrEK=9#)SA*scY1R4`YsHe!Mt!!TBJU@`wb-BJ-)3%;JigMdC$MX6_EZiph0*7Ch-% zyhd;{neSB8FyxCyHnpiyWP56Wf_idQ&w6Wwzoo$BRIq0lLm zyw{tk?=2)I#~eL;0%#N})q~>YLG=!&(&#jAI(75Rf;n<-Bmh8gz)t`2tp|Vm7tW_U AMgRZ+ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_weather-storm.png b/Media/Themes/Umami/Icon/status/stock_weather-storm.png new file mode 100644 index 0000000000000000000000000000000000000000..968a077124e183d981c3485dad0deb102e1c04be GIT binary patch literal 1598 zcmV-E2EqA>P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY3ljhU3ljkVnw%H_00kFGL_t(Y$HkRfY|~X7 z$G_+NFYUT+TsBq$?b?lj8)Hn!kkMj65CTz@W?A~6i4z`-Au&cs0D007F~W-%V!{)= zC=+It2S8zA*%$)b*kD6o>Xxy&b(hk1|9}6z{Qo`22NtrMs4>Rh%gMQYKRMr%^F6?S zT;%U8s`gho1VK2?7~2Q{l+uQDI=$yiZC(Gr22}g2?2;sPZQQUi%jtAcO^c!Xm+o|X zd%K!Wr&pY*tsD49fSQ^bvzAcari8x73E`Zx@l7vmDtWei6;)LQs;WYbsd&(RpFOzW zeUB0{hZDj%OCPhCvre2mc``9sPn-_W658(amE|9=t8~kXs-UT(>E%Ia!ZR2T+RrZqIso>`@5K*P1anG(;|3 zIFFv*UL=!Aq*6L0lMG1~;q`7Hf+zq0ik2;tiHLpE^@Pct9XobdL`itx?y#G$T)7PQ zIu9x;D!{?f&~ODgIXN(!&9LR!kY&w&d}2+*xH68>kzouD2GeUQp8LF`qr;d1sPIYe!6+{mQrVaKCa)m0f!?WCX$=%R>DEaW@##b~5!KY-&?0M-7gBF6Z(f&v#LQNorNw*UYVi8xYv3O3t9 zIGqJBnPh~+VdUiIptGwJJ74kPQ8a|Mo2?K;0VY`n2l0jLghTXj%-x;1EKgF+`&gI2;bN-@XN_B@=~3h3M_+#hp8y;0$Bc{wmiL z0O$Or@|ER;QURJ4Ln@U5r3A7p!(y>OQ4|Q2BBkpPB?*fbEynVaV*EZZ0A1I?IpA@7 zuxrAMJYWx#UesT-N3-W>&3+->7y|(4dI}dW{-Awx`fTQmsNlpnmY$onnmD&h#^BeG zdK$?*5S*$eAbbdh(2YLlK^WqksrK0&PwczZ-o^9^yY9z>#_r`XXOE%|3&3ewHzI5SWyvM!>t8gI@x;bz+DB0JE0~wDc{? zfBHZr<700~zNT-DfVYrm%)1TuK`ZWll_0^6*8l_n5YgAjCIg@PAEeHCI{2D5%hBJW z(ZQ}G($40?M!;+0)Mzz|D@p`ui9m;goF^Y~gnteakTs&Oad5^%o2`t1cRl786(aN? zNryVmNjsYlPfQ&Kmx{#xF`W+nro-tbTpx=O5Z`q1rvjM355bgeB|X<~6DHP7nmR-{ zpG%YhJJC9@85%4bO8QHBoogAU56d2{Hv(P@BlD)ue*m5*L%K-&LH+;$ z03CEiSad^gZEa<4bO1wgWnpw>WFU8GbZ8({Xk{QrNlj4iWF>9@001R)MObuXVRU6W zV{&C-bY%cCFfuSLFgYzSHdHY literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/stock_weather-sunny.png b/Media/Themes/Umami/Icon/status/stock_weather-sunny.png new file mode 100644 index 0000000000000000000000000000000000000000..7f6e10d45a4594319dbadcc6ddaf72fe71a72cdb GIT binary patch literal 1079 zcmZvUc}$ac7{{M#aS9lDvDgf-WKKPMoXnw5VdLbx5#S3ILnUy?!kKJga=%4gl-~QJ4roTWE!H3jp9Kf~8>qSvvu;zX4bQfb21V zObTE^2EZ=@*pyy!FX|%z*pMC^lL265V{R4{muv=Ll}8?k4gvtI6a25}Isx7i(6s`r z^+&DP%-j;9R$x}!Nz@5QhX!>5)Cp1Rk2-(z-MkA>`)1Y$n7>*9HU(gV5bK3_TlkhY z>_dZuzAHtYfPNsQ+f&Wq_eFR=20Kn-R}Mar(Dg$4b^zTVq#H%_!wCA3n0}N>cb%ZS z6X~7|Yz#2RHi__V0NxW}+n4xQgI&4U6N0@V*r&k1lW;H$2S3K412`0hL$Npu9FM^9NF0yCaVd@;#EDp({0t}MI2n(|1T?1NR2ohl!>J6MK8`aQoH>hg*T|p6 z!g(|X8O%`g%D=m3d zPZpcW(ywIcH^!_%X1Rq~Zew1*U|tV1D}Bt$7(@3n^Z-K-GxP|9Mh2%Dvcxb;tE;PL zJ7ymMfR|bteh}8|^xEw!5JzYP^?^0e}@3DiP;@VbvW7z}g-n7Dc;sDP%vg(>L^AxbTTgZ1YmK?%V49c}7lg?pn`> zH{JF6W=@5ChoQOIv8p5giDOl{qjTKYz|AiA%4+8`O8axVwB#&>B=F4nnnug_bskdJ zh8HzGr9s2Ji#(TJtJ2KO#%JG;I4Q5pZxe6%>$G+8;pfH4dG(4LeLd`$p9a=g=I!ge z`d#KBJB!@kE&fA#KYiHqR<1O99PPTul+ROg1sI9hwz3AZePHZ*h);f4r#PZCO zu7l$OUg#d#rQHCqRUP!HIxsCIMU|?Cz+)=8TBYQCm84GOL?@~^l8~^-8KqMy063XJ zrChDzE8bARqj;1D<>x_pM{|AoUVHdlZ+D8~Q`F^oU+#YlDQU{2_!GdRxLgm)+k@*9 d&E@fVK76jyH8egan0)Yxl}D{{RVZHKYIl literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/sunny.png b/Media/Themes/Umami/Icon/status/sunny.png new file mode 100644 index 0000000000000000000000000000000000000000..7f6e10d45a4594319dbadcc6ddaf72fe71a72cdb GIT binary patch literal 1079 zcmZvUc}$ac7{{M#aS9lDvDgf-WKKPMoXnw5VdLbx5#S3ILnUy?!kKJga=%4gl-~QJ4roTWE!H3jp9Kf~8>qSvvu;zX4bQfb21V zObTE^2EZ=@*pyy!FX|%z*pMC^lL265V{R4{muv=Ll}8?k4gvtI6a25}Isx7i(6s`r z^+&DP%-j;9R$x}!Nz@5QhX!>5)Cp1Rk2-(z-MkA>`)1Y$n7>*9HU(gV5bK3_TlkhY z>_dZuzAHtYfPNsQ+f&Wq_eFR=20Kn-R}Mar(Dg$4b^zTVq#H%_!wCA3n0}N>cb%ZS z6X~7|Yz#2RHi__V0NxW}+n4xQgI&4U6N0@V*r&k1lW;H$2S3K412`0hL$Npu9FM^9NF0yCaVd@;#EDp({0t}MI2n(|1T?1NR2ohl!>J6MK8`aQoH>hg*T|p6 z!g(|X8O%`g%D=m3d zPZpcW(ywIcH^!_%X1Rq~Zew1*U|tV1D}Bt$7(@3n^Z-K-GxP|9Mh2%Dvcxb;tE;PL zJ7ymMfR|bteh}8|^xEw!5JzYP^?^0e}@3DiP;@VbvW7z}g-n7Dc;sDP%vg(>L^AxbTTgZ1YmK?%V49c}7lg?pn`> zH{JF6W=@5ChoQOIv8p5giDOl{qjTKYz|AiA%4+8`O8axVwB#&>B=F4nnnug_bskdJ zh8HzGr9s2Ji#(TJtJ2KO#%JG;I4Q5pZxe6%>$G+8;pfH4dG(4LeLd`$p9a=g=I!ge z`d#KBJB!@kE&fA#KYiHqR<1O99PPTul+ROg1sI9hwz3AZePHZ*h);f4r#PZCO zu7l$OUg#d#rQHCqRUP!HIxsCIMU|?Cz+)=8TBYQCm84GOL?@~^l8~^-8KqMy063XJ zrChDzE8bARqj;1D<>x_pM{|AoUVHdlZ+D8~Q`F^oU+#YlDQU{2_!GdRxLgm)+k@*9 d&E@fVK76jyH8egan0)Yxl}D{{RVZHKYIl literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/trashcan_full.png b/Media/Themes/Umami/Icon/status/trashcan_full.png new file mode 100644 index 0000000000000000000000000000000000000000..a6b9cd3b5498f66c6a0caa0481b70399a2036e91 GIT binary patch literal 1589 zcmV-52Fm$~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00k0BL_t(Y$F-GdY*bYg z$A52~c{9_!6d@I{g$hLxLlh7dQ9(e7F$Oo{7h_B`Bx>SEzxmz7#2Dj(i62l>O$Za9CUbyJgL~CtNo}vG-^+HdZ@J?&^*l<$iDECGeZH4SeT6|i zrr{|kGx=H&R9@WQ6UB*Ndj`Ps>RxSa=vw*s&gW@wZzq?{a^L+AaP*f&EM2mc-r})z zpJ`dxwnGqx%$wK7!UfB5-IZJ%97^8Jk<#g1A)o&W(e7lwHGTV5!!X{kjFtBKb+^*h z*$G;cah%j)cXu~!^XJpt)WWjv>ycU$g&|T(8XKFbl*`p@Zz(_wf~}q_cAgYu2vCvTd?n zj=S&K!pRfIak9M)vj9@c+`U`YGrz4nrEeJqdM5u`r&2JqzRaelZJ&I4h+^+CmUXWn ziXwF4)UgPoTHP!_9MyKc@%EE@q!d@I^NhKkdp2*QTJ;Hn8sC2XC6#KQ4&}rPm&x?0 z=klXlw{B?3xEX}h$Ycsd6iR|n5=AOWAhpSuNnbEo69hGAZrH+Yw{PUf9}aWw+#i(7 zr^z?|)W7@nv89*E6pZmWqQ&zp98=;r3ARZlYvKtT5i7KIG00L2e1cky(Gi2TmJSYo z_YFhC-;%43$v2EKUI_ZH&e-OH9K?YB_DwX$rK`Oce#%&@9MiC;$G-CYPD8r>9 zQQ}xcU^q0^FJ!oMhK7Q|pBzW3I2A;q`ieh+RR6|6F*07hq|=0}7^x&-JkD_G7)?z} zsa8sa!8sPUufs50yzD$gEEpK|`NvN`Stl*F0MxG7`_%A~k^&~Cpa1{(6MBoS^_w}4&Hw-aC3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9bV6J literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/user-trash-full.png b/Media/Themes/Umami/Icon/status/user-trash-full.png new file mode 100644 index 0000000000000000000000000000000000000000..a6b9cd3b5498f66c6a0caa0481b70399a2036e91 GIT binary patch literal 1589 zcmV-52Fm$~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00k0BL_t(Y$F-GdY*bYg z$A52~c{9_!6d@I{g$hLxLlh7dQ9(e7F$Oo{7h_B`Bx>SEzxmz7#2Dj(i62l>O$Za9CUbyJgL~CtNo}vG-^+HdZ@J?&^*l<$iDECGeZH4SeT6|i zrr{|kGx=H&R9@WQ6UB*Ndj`Ps>RxSa=vw*s&gW@wZzq?{a^L+AaP*f&EM2mc-r})z zpJ`dxwnGqx%$wK7!UfB5-IZJ%97^8Jk<#g1A)o&W(e7lwHGTV5!!X{kjFtBKb+^*h z*$G;cah%j)cXu~!^XJpt)WWjv>ycU$g&|T(8XKFbl*`p@Zz(_wf~}q_cAgYu2vCvTd?n zj=S&K!pRfIak9M)vj9@c+`U`YGrz4nrEeJqdM5u`r&2JqzRaelZJ&I4h+^+CmUXWn ziXwF4)UgPoTHP!_9MyKc@%EE@q!d@I^NhKkdp2*QTJ;Hn8sC2XC6#KQ4&}rPm&x?0 z=klXlw{B?3xEX}h$Ycsd6iR|n5=AOWAhpSuNnbEo69hGAZrH+Yw{PUf9}aWw+#i(7 zr^z?|)W7@nv89*E6pZmWqQ&zp98=;r3ARZlYvKtT5i7KIG00L2e1cky(Gi2TmJSYo z_YFhC-;%43$v2EKUI_ZH&e-OH9K?YB_DwX$rK`Oce#%&@9MiC;$G-CYPD8r>9 zQQ}xcU^q0^FJ!oMhK7Q|pBzW3I2A;q`ieh+RR6|6F*07hq|=0}7^x&-JkD_G7)?z} zsa8sa!8sPUufs50yzD$gEEpK|`NvN`Stl*F0MxG7`_%A~k^&~Cpa1{(6MBoS^_w}4&Hw-aC3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9bV6J literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/weather-clear-night.png b/Media/Themes/Umami/Icon/status/weather-clear-night.png new file mode 100644 index 0000000000000000000000000000000000000000..e7595f7c1e0fd9d471428bce71ffc22c62e1b327 GIT binary patch literal 1079 zcmY+6c}&v>9L2wEx`BkLGpN}WqCzwb{Iy2V&TIq9pfCtW|JX9-Kw7{|4zYmKjX4E{ zUR*|;Vj{^{@Sva!bjz|pX-ivrOh&;%4|@E1P}<+Gv`{EzES0~OeBK}L<-OOOlbsf^ zN=K;X}ph8~I594?N}#JpOIy>Co`tgiI=vzd|9ED<kBv)&Vu?^XF)p1DOJ(B|aXqV zCY`~oGn(}#^SlY2H=za-YBZw;Gio$jjHuOQv6(D3v&Dv5t*F&*w%Spf-C|p?*c}$T z!)kZf78Y#|r`_RPSadEpT@I&f(dlx!utf}WVYmy!G0g44-5Bo1@g)rRU~V_=Uc%f< zxZ8s-dGIB#+vD|kmp$HPug|yQTV3(3`c{LiuCA@F1zB73ula-c{pH7q`XK-uW#nb&!)wvi7iwzjE?#T6&SZ{^@&!WCxJ2q+@_4*nuXlNQWyKe~ z_WJ{Yjg5b&cKuWW03l~G)5x8cig)_~Lf)W|lcjlJ}IDbB1G` z7*CZ?D<$vVms|dAJX=C3zt^Rs8l2X^>uJ!=H`;62t7%7$I<@5>$ zq*T+37<5`x-C0Ii6t#>Vl}5?R!D!JH01zE4(ux>#Qt_)65D_90hn$Q<5~zd|q+`cP zgoIcGAtA^u&k4f+2IZBsvnAgF5g`!bkc2owB9%ZS5fe#-lUoDybm(aS07_~$nU`F6 F^*@l(TxI|O literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/weather-clear.png b/Media/Themes/Umami/Icon/status/weather-clear.png new file mode 100644 index 0000000000000000000000000000000000000000..7f6e10d45a4594319dbadcc6ddaf72fe71a72cdb GIT binary patch literal 1079 zcmZvUc}$ac7{{M#aS9lDvDgf-WKKPMoXnw5VdLbx5#S3ILnUy?!kKJga=%4gl-~QJ4roTWE!H3jp9Kf~8>qSvvu;zX4bQfb21V zObTE^2EZ=@*pyy!FX|%z*pMC^lL265V{R4{muv=Ll}8?k4gvtI6a25}Isx7i(6s`r z^+&DP%-j;9R$x}!Nz@5QhX!>5)Cp1Rk2-(z-MkA>`)1Y$n7>*9HU(gV5bK3_TlkhY z>_dZuzAHtYfPNsQ+f&Wq_eFR=20Kn-R}Mar(Dg$4b^zTVq#H%_!wCA3n0}N>cb%ZS z6X~7|Yz#2RHi__V0NxW}+n4xQgI&4U6N0@V*r&k1lW;H$2S3K412`0hL$Npu9FM^9NF0yCaVd@;#EDp({0t}MI2n(|1T?1NR2ohl!>J6MK8`aQoH>hg*T|p6 z!g(|X8O%`g%D=m3d zPZpcW(ywIcH^!_%X1Rq~Zew1*U|tV1D}Bt$7(@3n^Z-K-GxP|9Mh2%Dvcxb;tE;PL zJ7ymMfR|bteh}8|^xEw!5JzYP^?^0e}@3DiP;@VbvW7z}g-n7Dc;sDP%vg(>L^AxbTTgZ1YmK?%V49c}7lg?pn`> zH{JF6W=@5ChoQOIv8p5giDOl{qjTKYz|AiA%4+8`O8axVwB#&>B=F4nnnug_bskdJ zh8HzGr9s2Ji#(TJtJ2KO#%JG;I4Q5pZxe6%>$G+8;pfH4dG(4LeLd`$p9a=g=I!ge z`d#KBJB!@kE&fA#KYiHqR<1O99PPTul+ROg1sI9hwz3AZePHZ*h);f4r#PZCO zu7l$OUg#d#rQHCqRUP!HIxsCIMU|?Cz+)=8TBYQCm84GOL?@~^l8~^-8KqMy063XJ zrChDzE8bARqj;1D<>x_pM{|AoUVHdlZ+D8~Q`F^oU+#YlDQU{2_!GdRxLgm)+k@*9 d&E@fVK76jyH8egan0)Yxl}D{{RVZHKYIl literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/weather-few-clouds-night.png b/Media/Themes/Umami/Icon/status/weather-few-clouds-night.png new file mode 100644 index 0000000000000000000000000000000000000000..bb3f5c1429288daab75290c317bc7afb57168da4 GIT binary patch literal 1269 zcmV004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 E3neQvFaQ7m literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/weather-severe-alert.png b/Media/Themes/Umami/Icon/status/weather-severe-alert.png new file mode 100644 index 0000000000000000000000000000000000000000..ab23f36adcef8a4f3d4e18e7074daff4ecdf105c GIT binary patch literal 1440 zcmV;R1z-A!P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 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 V1|9}E22KB4zg>HMs&`P5{sQ;N?J58O literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/weather-snow.png b/Media/Themes/Umami/Icon/status/weather-snow.png new file mode 100644 index 0000000000000000000000000000000000000000..18e9843178a3e139189aa5a96927332cfc4318f9 GIT binary patch literal 1507 zcmY+83rv#-5XTRlo5+Z6zKGKd9XcmL+cH7R96@1#6NAF05@jNw2;(I%r}4$KwBNIZ z2|7VkWD_jn3qe6jTM8nyls;&M(o*^WERTYM%RpplS(bSD-R16b_rF6zcLqAoTQ&~> zoP&e>cLRWn&F9Vr0ClOP82~;w;uqovaG&k;Hfjz4xJC!<4gom12H@NUfC&IN*AI|E z0k9kdpkD@9no>}`>l*-YObObT2H>>N<_>VBzy)AdY3Ppd005Ytp3ce1AxRR$Fc!*q z57vb&G;5}VqFD%*h2TsC%R+Gm6JfBk33fKl&L&8XEpl>*na)hf&c-pEg%NiCPi9Jc zB-t2>Gf_MXBUlK|Zt15VPwE_X}d2>^9 za}x_?5F|pfQEpkuy$9vByr=c`0=c}6g)&e)3&olC>j+LDxcxN2#z+pv$ssQ0W#`(+ z$-R`5d+GPQ%NPI1&%IJ`IsfYATP4M}xh34P5^j0vox9xfyWEEr_a8pGUs?62s;Z*u zN#&F3%IccOH8qcGYioG?r*-_=IzEpt;0XkE0wGTztP=?NLSday$QRZNgd%>uNLb$> z5Q&7M27#zS*w9el_)OH;)X>!2_`Lbq^OmL-akE&`B54)3wo0T@iL|X%Dr;@CA#IaO zWj5Mm3YlCXQz+yLrA(oeE87+AHk9ouMY~F=QY%&JcD1@){X+GkL)~HHg+|k<>FVs# zc51a+S3B>+kQgSYBENECU0Ufx&^n!GXb{!J(nSVe7DU#5!VQWOQ_7^wsF= z*RRIM$0dOxD*zly-@P*o96oY%TDE+}%2jk9A79^q_@vad6Q@s~J)3!fg<|Z&YehFp zZ{NB5uu3YE%a!VmPEGgBa|WY%XxM7Cj*Pr|JvKHz{^rd;6B84YlW*Tn&DgvDtib*F zCjj7Z==(r_hv``!Isk{lV1K{xwbJN=uAkrI`qadwT*qj%2b;eu)P2?~YHVV;iQVF& z{#=pzxBQ03jp`*oZWX6EI@Gxb9+DoYITaUhWGeRyoiN4mCRfwqxlUEkS#z|JNMJ7A zS1;Q4vubIwuW!}CcQT(N8l{jf0;B1 zh%oND%16D${pY4$W(J$+zh^GIm+X!{sG58qd{xy6_M)!t0!^*#4R&OsKB{Yhed zrEK=9#)SA*scY1R4`YsHe!Mt!!TBJU@`wb-BJ-)3%;JigMdC$MX6_EZiph0*7Ch-% zyhd;{neSB8FyxCyHnpiyWP56Wf_idQ&w6Wwzoo$BRIq0lLm zyw{tk?=2)I#~eL;0%#N})q~>YLG=!&(&#jAI(75Rf;n<-Bmh8gz)t`2tp|Vm7tW_U AMgRZ+ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/weather-storm.png b/Media/Themes/Umami/Icon/status/weather-storm.png new file mode 100644 index 0000000000000000000000000000000000000000..968a077124e183d981c3485dad0deb102e1c04be GIT binary patch literal 1598 zcmV-E2EqA>P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 zVoOIv00000008+zyMF)x010qNS#tmY3ljhU3ljkVnw%H_00kFGL_t(Y$HkRfY|~X7 z$G_+NFYUT+TsBq$?b?lj8)Hn!kkMj65CTz@W?A~6i4z`-Au&cs0D007F~W-%V!{)= zC=+It2S8zA*%$)b*kD6o>Xxy&b(hk1|9}6z{Qo`22NtrMs4>Rh%gMQYKRMr%^F6?S zT;%U8s`gho1VK2?7~2Q{l+uQDI=$yiZC(Gr22}g2?2;sPZQQUi%jtAcO^c!Xm+o|X zd%K!Wr&pY*tsD49fSQ^bvzAcari8x73E`Zx@l7vmDtWei6;)LQs;WYbsd&(RpFOzW zeUB0{hZDj%OCPhCvre2mc``9sPn-_W658(amE|9=t8~kXs-UT(>E%Ia!ZR2T+RrZqIso>`@5K*P1anG(;|3 zIFFv*UL=!Aq*6L0lMG1~;q`7Hf+zq0ik2;tiHLpE^@Pct9XobdL`itx?y#G$T)7PQ zIu9x;D!{?f&~ODgIXN(!&9LR!kY&w&d}2+*xH68>kzouD2GeUQp8LF`qr;d1sPIYe!6+{mQrVaKCa)m0f!?WCX$=%R>DEaW@##b~5!KY-&?0M-7gBF6Z(f&v#LQNorNw*UYVi8xYv3O3t9 zIGqJBnPh~+VdUiIptGwJJ74kPQ8a|Mo2?K;0VY`n2l0jLghTXj%-x;1EKgF+`&gI2;bN-@XN_B@=~3h3M_+#hp8y;0$Bc{wmiL z0O$Or@|ER;QURJ4Ln@U5r3A7p!(y>OQ4|Q2BBkpPB?*fbEynVaV*EZZ0A1I?IpA@7 zuxrAMJYWx#UesT-N3-W>&3+->7y|(4dI}dW{-Awx`fTQmsNlpnmY$onnmD&h#^BeG zdK$?*5S*$eAbbdh(2YLlK^WqksrK0&PwczZ-o^9^yY9z>#_r`XXOE%|3&3ewHzI5SWyvM!>t8gI@x;bz+DB0JE0~wDc{? zfBHZr<700~zNT-DfVYrm%)1TuK`ZWll_0^6*8l_n5YgAjCIg@PAEeHCI{2D5%hBJW z(ZQ}G($40?M!;+0)Mzz|D@p`ui9m;goF^Y~gnteakTs&Oad5^%o2`t1cRl786(aN? zNryVmNjsYlPfQ&Kmx{#xF`W+nro-tbTpx=O5Z`q1rvjM355bgeB|X<~6DHP7nmR-{ zpG%YhJJC9@85%4bO8QHBoogAU56d2{Hv(P@BlD)ue*m5*L%K-&LH+;$ z03CEiSad^gZEa<4bO1wgWnpw>WFU8GbZ8({Xk{QrNlj4iWF>9@001R)MObuXVRU6W zV{&C-bY%cCFfuSLFgYzSHdHY literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Icon/status/xfce-trash_full.png b/Media/Themes/Umami/Icon/status/xfce-trash_full.png new file mode 100644 index 0000000000000000000000000000000000000000..a6b9cd3b5498f66c6a0caa0481b70399a2036e91 GIT binary patch literal 1589 zcmV-52Fm$~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000fw000fw0YWI7cmMzZ24YJ`L;(K){{a7>y{D4^00k0BL_t(Y$F-GdY*bYg z$A52~c{9_!6d@I{g$hLxLlh7dQ9(e7F$Oo{7h_B`Bx>SEzxmz7#2Dj(i62l>O$Za9CUbyJgL~CtNo}vG-^+HdZ@J?&^*l<$iDECGeZH4SeT6|i zrr{|kGx=H&R9@WQ6UB*Ndj`Ps>RxSa=vw*s&gW@wZzq?{a^L+AaP*f&EM2mc-r})z zpJ`dxwnGqx%$wK7!UfB5-IZJ%97^8Jk<#g1A)o&W(e7lwHGTV5!!X{kjFtBKb+^*h z*$G;cah%j)cXu~!^XJpt)WWjv>ycU$g&|T(8XKFbl*`p@Zz(_wf~}q_cAgYu2vCvTd?n zj=S&K!pRfIak9M)vj9@c+`U`YGrz4nrEeJqdM5u`r&2JqzRaelZJ&I4h+^+CmUXWn ziXwF4)UgPoTHP!_9MyKc@%EE@q!d@I^NhKkdp2*QTJ;Hn8sC2XC6#KQ4&}rPm&x?0 z=klXlw{B?3xEX}h$Ycsd6iR|n5=AOWAhpSuNnbEo69hGAZrH+Yw{PUf9}aWw+#i(7 zr^z?|)W7@nv89*E6pZmWqQ&zp98=;r3ARZlYvKtT5i7KIG00L2e1cky(Gi2TmJSYo z_YFhC-;%43$v2EKUI_ZH&e-OH9K?YB_DwX$rK`Oce#%&@9MiC;$G-CYPD8r>9 zQQ}xcU^q0^FJ!oMhK7Q|pBzW3I2A;q`ieh+RR6|6F*07hq|=0}7^x&-JkD_G7)?z} zsa8sa!8sPUufs50yzD$gEEpK|`NvN`Stl*F0MxG7`_%A~k^&~Cpa1{(6MBoS^_w}4&Hw-aC3Hnt zbYx+4WjbSWWnpw>05UK#FfA}SEig7zF*rIiH99diD=;uRFff+$I57YK03~!qSaf7z zbY(hiZ)9m^c>ppnF)=MLHZ3tXR53C-GB-LgIV&(QIxsNhUI<MzCV_|S*E^l&Yo9;Xs00000NkvXXu0mjf9bV6J literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Pointer/alternate.png b/Media/Themes/Umami/Pointer/alternate.png new file mode 100644 index 0000000000000000000000000000000000000000..6465e7ed321d2d08dfb98c050f20f8ed25a41f52 GIT binary patch literal 922 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 Cg3Ad2 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Pointer/help.png b/Media/Themes/Umami/Pointer/help.png new file mode 100644 index 0000000000000000000000000000000000000000..57327a4c5fc2c9ba406574a189bbf4bb5988d7df GIT binary patch literal 1346 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 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Pointer/pen.png b/Media/Themes/Umami/Pointer/pen.png new file mode 100644 index 0000000000000000000000000000000000000000..e836fe4886861df5aa0ca6a47bd09ef2a7a0a20b GIT binary patch literal 1253 zcmY*ZeNfVO82^f>L`McpiVm)}iWy1%P)?CrmZ{}RO=C;TKTy+hw_Vn>lCC>iPJ2<@ z;e263TY^*wmb%!|>_=gU#WdTPi4JtTL**qAlOblm9rf2fchBAPa?j`a-1FQ$R}QA_ z#bMWB0RXs^eTkU>fDL}b1OfmOWmk0o0QB+Qsk;H_s9E_YH^^Vi--E*u)pd1st*xyp zm8zzuMy*!&_V!j*RtkkesZ`q3)FcoHs;a7LYipHCrA#K{^Z7cRP9ze|&dwGW7Z(*3 zojZ4~wY9apyu7osb7^VGX0u5olCrY0ii!%oUfz!oq^r>m3^# zb2^=agM%iM$?0^uT&~&KS(nS@a5yF=CQK%i*=)Ak?Jk#VZfFMc0p-?0e-MDe1 zrKLroP_(tRy+6(9=;-+PxW!^|xm*Z>NKap10|2;Gl9_f0_%#FuqR~PA`54R!EEc;G zhg%gI8oGM*YCN7mB!-2N)~qFy$?M)}S-*b6J24-JhlhvLBN&m9P;_)Oi@klv&iMGO z>?22W^Ye?oxKw$qS*cR>K7`>>v)MW|Gvj>y`t`zs$K&z*>xXx7adFYNw7k6h_NTKK z=>Bcw?cbZ&HV|+;-0!C(?&h+q^Le{KGOdtQNTUVk2(E~muGRcl8`glOkp`y??8s9) z&rs-%aeL-zu^$A7V|a%j^-X58RQF3i7hYn;t|WyWP=0#5t^T)}nMmAWVxE?(>3d8) z`eMh~#Rqa$ECU~L2Ku_UJvm;h#tHB!o7ML+j)}s`Au{%`rb@Q9uR+Gk+N$N(`s4k^)Zt>4Myn0uTj7K@2E{0mX8eOipwxCpv}!u{aR) z55Kl!P&uPTmx~#!;AqszNl1A+5+!_CQEiD&!79bub+2M zLw!|ha!xV;q*eTr%lP=XMx)W`bfcrAGcz+IBO_yD zV^dR8IF6f4rvCnZv)SC&*QZb@1_uWbLg52kE>|!Z#BqFkd)wu5t*)-FtgJYlPP5s( zv9aNHyPZy_-EQAH!0Ywy?Ce-97Qf#=Jw5I7`TTyrMx)u?-30EDb+0bg8V+4|#`(RCt-2E9Q;q2!Ly5pO}AP&bN=Fg4PxH}8<_GG2k|M1H$e*WBG;=Y`AXUfJr5 zdvL8_edys}c-Qacop1X)GkJz`^{w}1l8*dG_StBf_osqOi@EBWko4HA*WKFR@Gl-d zxnXgKZVtW9=j+lx{O3W`x)ga5Z9bKt`t!?2nIkQ^M_=@e({p&)?$e6N;{)xtv6i8W z6G<5pX9X8)yRQz6y+b^93+3$-WtVyPds^$40e~*Ad{-`&$YtENk7W|M3<|MAjEQlW zSh1MJ;uaNii#SY-&Bd@y<=tEV6E1Z~FP!@n3Ncm*6Js-3CE`LBH&$gI3dO66yRkJ@ MSJn%rDq64q3#$^;@Bjb+ literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Pointer/text.png b/Media/Themes/Umami/Pointer/text.png new file mode 100644 index 0000000000000000000000000000000000000000..f2b17c25a6efda34d527f325ebf2e87b249ab788 GIT binary patch literal 5514 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 bMi*85OJ_YCc_3J(_`CIR@ur_~TDSdwe`1@| literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Pointer/unavailable.png b/Media/Themes/Umami/Pointer/unavailable.png new file mode 100644 index 0000000000000000000000000000000000000000..fcb8d5b7f772352e11bcd92ee254b673794cfc70 GIT binary patch literal 1314 zcmY+=c}&v>6bJBE6^bBgER<4oN?`(qVmSn?3Y0@2Qec&AF6wAGgNh&`S_uGaGHG-gfQ)Omz_MBjQ)3u(fa2`{oU;J)0Km}!oS^_r zF#)K20H@sM?r-)2V4f$&B}D;%#bV)bI2MZqL6H9bez{z(P$&=tQ7V<4ot+|)NG6ks z#bN|OhKGj{1Zin$;qiDPkw_pA)YsQvx^$_ou1=v)peWkW(b3x4Dv?MkDk?Y}j?ri= zC@3f`Ep2RUY-?-d^ZCWa#Zsx1&1Q?m;+&kEo}Ql3(b4AS=8}?1n-QKRrE-qUg-b%=q~D*w~m>tJUdrCX>l* zHqXw^8Vm-b(P%cCZ{NOcHk&09iBu|WZ*P~$WL;fd@7}%BXf!M7%1|zsn@pyrrY1h0 z&*SmU@%rzR@TFMbi85xctF5p$YuDP^*{|QQVUr^s?@V;rym^Z&iR9+K-OJ0% zYlpWtne5}^6Btb0vo9k0KwQQ7S{`4}(jpSK{?>Nm269*N@K2>u+0)zq*Q-~rUk?lp zzE!D)4ThyZy?_7VKOdKT`ZPbkeCp!jm*+TCzjSEIf#^t#Wfge@05cdHNlOauL+j|r zRbd;?XBKFW#(qKKT+;g1$LD#CF+9(Pm6w%=d3m)yBX<2lM>XzpN6DR=i=I;4fspqj zAKFX!yrRtWSk}%p?)TKx+(f64-yLd6)XZl8X#7mhm70gKFNh=Ko%-whj}M>1iXZn4 zToLXl4LrEjgWG$#m&Ts^Zf5_)9nYiRx3|qP)*iMqK&)w=RP)SMfLDo}G(=l=jF4k~ zob{B}QCx1?#jDM#$wP1Li`+(+J_+Ew5%-~tk4InH1>QI#uMVD!A$Zf*oV?sBzj*qI z;PuljHkwJB%bnO`qE4BIQ@8y!-)n`GjYLnI&{Fm8I=j$cFKoLTot6DF3qzCTZWIuh zBtp;q3y1uNeyHvcUiYb@uWLK~H;xe${OFu+XJ-yR1!rmksbtTQ>P;H9pkZ+T3Bn}5 z-2&{OEMd#Q?5#%)$?K{Pr{TtGKUvX`GiYL%A2a)9wtL96QFp5&KX#WVbyFK&)<3jU zTvX+_6op_L*f--Qk6Gt39k17ZI~cgr^$1yU>zp?|JuFw#weCdRi8uEXLK6nUdu^SE z3}L$yKJqSI5crFxnvPD&sgm^ zHbsgBfUonT4&|q@^3$oQdFibDbnv72QOJ})G9@U<*O%%aMD-6OQv#?I%4l)Z#s4j2 t=cHw26oDVbH<(NbAo~U<`T0^QyQq|am4!E3ziI;jh>ePml!h~Z`WNXckDdSk literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Pointer/vert.png b/Media/Themes/Umami/Pointer/vert.png new file mode 100644 index 0000000000000000000000000000000000000000..9c4cd257f8245b91fa716fd0830534dfe9fa1acb GIT binary patch literal 1084 zcmY+?YfMuI7zW^vg5^@~S3AwXAmB1fPRliQk=B(7K^S0llQ@MT_YDUY=0HMMkjPjZ zW5EhI7_Gyk3fLK%aI-3ft+4?%1})=$=_yd)(9^I&VVnirpFPR@>wDjSUuAsUSqHnf z>;S+aI*OAB06zP%u>=6l!*2!vz$$|s%Lb5sN7+p|w0}&EN{j_4@(19T1DFATzXVW# z0IVbfFz*9+ees|x;XMG0viNh{NB|%Rg3sp@1R;?~2!fEyfR4R!?GCDd+5QI!7 zo0yoGnVDHwSWu}{D2jG=cFN`Q(a}+DwRqmlMN0I zHZ(NU*48#PHHpRI)&m_K9sT|NLZPs|y9uN~Of*uXGsdU{%~*P|%9 zwzjsivNAk8JT*0iqNvemR4SE2Lqlq{dUA4detur7)nXWiVH;*Vi{7Ab?J%69hr!DDD72 z0WUG`JXnxOmX=mlhsb2IwY80{ogKyA-oe4a(b2`t&E4I@!|U*2>QQfRZyz6@W4^w= ze*P!kJ$W*a&Ik$$4r88XM<#GDUe5UJMnUnd((>}k%DdI~8=6Ir9=Azy<;)yRca)od4d;2w&)!kxhG%OfyUCzJcJUd%qnn4EIAGt-k>U?aHt%o9>Hk%q; zLSB#zY>Z7Vedk%wPj@=2NcHEIyYAA*3jRsCU_52};t1_J<7>{BF&B>64);8%tz&(7 z<7aI-B%KM|KqJA*J5K!)@tFJm;Q0sd^(vaOC??A@(dV@jRpH6zM_9#Q4!b zv-k=J_L?WK~dXs)E@Q5Nc7YW_$t=ZnRs z(+gt7PKk!}{Ku^?yPVShnyVQf$nO4tT2&-)38~3YN2ihl%zK%erRspIHnGo76S2D3pc-GZAE==-#*g8+?BKT2}g3 mz(DBXG$fct59c!IOeBOE%y=`1xb2q-01zD+$7ws0Ech4vO(yFA literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Pointer/wait.png b/Media/Themes/Umami/Pointer/wait.png new file mode 100644 index 0000000000000000000000000000000000000000..6f7efbdad8343188d5efe6569530bde2d6bd4719 GIT binary patch literal 7107 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 + widget->color = Color(0, 0, 0); + if (!widget->font) + widget->font = Compositor.theme.font.sans; + if (widget->image) { + widget->width = Max(widget->image->width + 8, widget->width); + widget->height = Max(widget->image->height + 8, widget->height); + } + Rect2D(win->render_ctx, x + 1, y + 1, widget->width - 2, widget->height - 2, + Color(232, 232, 232)); + Line2D(win->render_ctx, x + 1, y, x + widget->width - 1, y, + Color(96, 96, 96)); + Line2D(win->render_ctx, x + 1, y + widget->height - 1, x + widget->width - 1, + y + widget->height - 1, Color(96, 96, 96)); + Line2D(win->render_ctx, x, y + 1, x, y + widget->height - 2, + Color(96, 96, 96)); + Line2D(win->render_ctx, x + widget->width - 1, y + 1, x + widget->width - 1, + y + widget->height - 2, Color(96, 96, 96)); + if (widget == win->mouse_down_widget && Mouse.left && + @widget_is_hovered(win->x + x, win->y + y, widget)) { + + if (widget->image) { + Blot2D(win->render_ctx, x + 5, y + 5, widget->image); + } + + Line2D(win->render_ctx, x + 1, y + 1, x + widget->width - 2, y + 1, + Color(152, 152, 152)); + Line2D(win->render_ctx, x + 1, y + 1, x + 1, y + widget->height - 3, + Color(152, 152, 152)); + Line2D(win->render_ctx, x + 2, y + 2, x + widget->width - 3, y + 2, + Color(216, 216, 216)); + Line2D(win->render_ctx, x + 2, y + 2, x + 2, y + widget->height - 4, + Color(216, 216, 216)); + Line2D(win->render_ctx, x + 2, y + widget->height - 3, + x + widget->width - 2, y + widget->height - 3, Color(255, 255, 255)); + Line2D(win->render_ctx, x + 3, y + widget->height - 4, + x + widget->width - 2, y + widget->height - 4, Color(255, 255, 255)); + Line2D(win->render_ctx, x + widget->width - 3, y + 2, x + widget->width - 3, + y + widget->height - 3, Color(255, 255, 255)); + Line2D(win->render_ctx, x + widget->width - 4, y + 3, x + widget->width - 4, + y + widget->height - 3, Color(255, 255, 255)); + } else { + + if (widget->image) { + Blot2D(win->render_ctx, x + 4, y + 4, widget->image); + } + + Line2D(win->render_ctx, x + 2, y + 2, x + widget->width - 2, y + 2, + Color(255, 255, 255)); + Line2D(win->render_ctx, x + 2, y + 3, x + widget->width - 3, y + 3, + Color(255, 255, 255)); + Line2D(win->render_ctx, x + 2, y + 2, x + 2, y + widget->height - 3, + Color(255, 255, 255)); + Line2D(win->render_ctx, x + 3, y + 2, x + 3, y + widget->height - 4, + Color(255, 255, 255)); + Line2D(win->render_ctx, x + 3, y + widget->height - 3, + x + widget->width - 3, y + widget->height - 3, Color(216, 216, 216)); + Line2D(win->render_ctx, x + widget->width - 3, y + 3, x + widget->width - 3, + y + widget->height - 3, Color(216, 216, 216)); + Line2D(win->render_ctx, x + 2, y + widget->height - 2, + x + widget->width - 2, y + widget->height - 2, Color(152, 152, 152)); + Line2D(win->render_ctx, x + widget->width - 2, y + 2, x + widget->width - 2, + y + widget->height - 2, Color(152, 152, 152)); + } + if (StrLen(&widget->text)) { + text_offset = T(widget == win->mouse_down_widget && Mouse.left && + @widget_is_hovered(win->x + x, win->y + y, widget), + 1, 0); + text_width = PutS2D(NULL, widget->font, 0, 0, Color(0, 0, 0), , &widget->text); + PutS2D(win->render_ctx, widget->font, + x + text_offset + (widget->width / 2) - (text_width / 2), + y + text_offset + (widget->height / 2) - (widget->font->line_height / 2), + widget->color, , &widget->text); + } +} + +U0 @umami_checkbox_repaint(Window* win, CheckBoxWidget* widget, I64 x, I64 y) +{ + + if (widget == win->mouse_down_widget && + @widget_is_hovered(win->x + x, win->y + y, widget)) { + Blot2D(win->render_ctx, x, y, + T(widget->checked, umami_widget_checkbox_checked_active, + umami_widget_checkbox_active)); + return; + } + if (widget == win->mouse_down_widget->echo && widget != win->mouse_down_widget && + @widget_is_hovered(win->x + x, win->y + y, win->mouse_down_widget)) { + Blot2D(win->render_ctx, x, y, + T(widget->checked, umami_widget_checkbox_checked_active, + umami_widget_checkbox_active)); + return; + } + Blot2D( + win->render_ctx, x, y, + T(widget->checked, umami_widget_checkbox_checked, umami_widget_checkbox)); +} + +U0 @umami_terminal_calculate_size(Window* win, TerminalWidget* widget) +{ + widget->size.cols = RoundI64(win->width, 8) / 8; + widget->size.rows = (RoundI64(win->height, 16) / 16) - 2; +} + +U32 @umami_terminal_256_color_lookup(I64 index) +{ + U32* color_table = TERMINAL_COLOR_TABLE; + return color_table[index]; +} + +U0 @umami_terminal_set_256_color_foreground(Window* win, TerminalWidget* widget, + I64 argc, U8** argv) +{ + if (argc != 3 && !StrCmp(argv[1], "5")) + return; + widget->color.foreground = @umami_terminal_256_color_lookup(Str2I64(argv[2])); +} + +U0 @umami_terminal_set_256_color_background(Window* win, TerminalWidget* widget, + I64 argc, U8** argv) +{ + if (argc != 3 && !StrCmp(argv[1], "5")) + return; + widget->color.background = @umami_terminal_256_color_lookup(Str2I64(argv[2])); + widget->color.background.u8[3] = win->opacity; +} + +U0 @umami_terminal_cleanup_strings(I64 argc, U8** argv) +{ + I64 i; + for (i = 0; i < argc - 1; i++) + Free(argv[i]); +} + +I64 @umami_ansi_16_color_table[8] = { 0, 4, 2, 6, 1, 5, 3, 7 }; + +U0 @umami_terminal_set_16_color_fg_value(TerminalWidget* widget) +{ + I64 val = widget->last_fg_color_set; + if (val > 15) + return; + widget->color.foreground = Color(T(widget->attr.bold, + gr_palette_std[8 + @umami_ansi_16_color_table[val]].r & 0xFF, + gr_palette_std[@umami_ansi_16_color_table[val]].r & 0xFF), + T(widget->attr.bold, + gr_palette_std[8 + @umami_ansi_16_color_table[val]].g & 0xFF, + gr_palette_std[@umami_ansi_16_color_table[val]].g & 0xFF), + T(widget->attr.bold, + gr_palette_std[8 + @umami_ansi_16_color_table[val]].b & 0xFF, + gr_palette_std[@umami_ansi_16_color_table[val]].b & 0xFF)); +} + +U0 @umami_terminal_ctrl_seq_sgr(Window* win, TerminalWidget* widget) +{ + I64 argc; + I64 i; + I64 val; + U8** argv = String.Split(&widget->consumed_chars, ';', &argc); + if (!argc && StrLen(&widget->consumed_chars)) { + argc = 1; + argv[0] = &widget->consumed_chars; + } + if (!argc && !StrLen(&widget->consumed_chars)) { + widget->color.background = Color(0, 0, 0); + widget->color.foreground = Color(217, 217, 217); + return; + } + for (i = 0; i < argc; i++) { + val = Str2I64(argv[i]); + switch (val) { + case 0: + widget->color.background = Color(0, 0, 0); + widget->color.foreground = Color(217, 217, 217); + break; + case 1: + widget->attr.bold = TRUE; + if (widget->last_fg_color_set < 16) + @umami_terminal_set_16_color_fg_value(widget); + break; + case 2: + widget->attr.bold = FALSE; + if (widget->last_fg_color_set < 16) + @umami_terminal_set_16_color_fg_value(widget); + break; + // 16 colors + case 30...37: + widget->last_fg_color_set = val - 30; + @umami_terminal_set_16_color_fg_value(widget); + break; + case 40...47: + widget->color.background = Color(gr_palette_std[@umami_ansi_16_color_table[(val - 40)]].r & 0xFF, + gr_palette_std[@umami_ansi_16_color_table[(val - 40)]].g & 0xFF, + gr_palette_std[@umami_ansi_16_color_table[(val - 40)]].b & 0xFF, + win->opacity); + break; + // 256 colors + case 38: + widget->last_fg_color_set = 256; + @umami_terminal_set_256_color_foreground(win, widget, argc, argv); + i += 2; + break; + case 48: + @umami_terminal_set_256_color_background(win, widget, argc, argv); + i += 2; + break; + case 39: + // FIXME: Set default foreground color + widget->color.foreground = Color(217, 217, 217); + break; + case 49: + // FIXME: Set default background color + widget->color.background = Color(0, 0, 0); + break; + default: + System.Log("Unrecognized attribute in ctrl_seq_sgr: %d", val); + break; + } + } + // if (argc) + // @umami_terminal_cleanup_strings(argc, argv); +} + +U0 @umami_terminal_set_col(Window* win, TerminalWidget* widget, I64 row, + I64 col, U8 char) +{ + widget->row[row + widget->scroll.y].col[col + widget->scroll.x].foreground = widget->color.foreground; + widget->row[row + widget->scroll.y].col[col + widget->scroll.x].background = widget->color.background; + widget->row[row + widget->scroll.y].col[col + widget->scroll.x].char = char; +} + +U0 @umami_terminal_set_row(Window* win, TerminalWidget* widget, I64 row, + U8 char) +{ + I64 col; + for (col = 0; col < TERMINAL_MAX_COLS - 1; col++) + @umami_terminal_set_col(win, widget, row, col, char); +} + +U0 @umami_terminal_set_rows(Window* win, TerminalWidget* widget, U8 char) +{ + I64 row; + for (row = 0; row < widget->size.rows - 1; row++) + @umami_terminal_set_row(win, widget, row, char); +} + +U0 @umami_terminal_repaint_col(Window* win, TerminalWidget* widget, I64 row, + I64 col) +{ + U8 char[2]; + char[1] = NULL; + U32 col_bg_color; + char[0] = widget->row[row + widget->scroll.y].col[col + widget->scroll.x].char; + col_bg_color = widget->row[row + widget->scroll.y] + .col[col + widget->scroll.x] + .background; + col_bg_color.u8[3] = win->opacity; + Rect2D(widget->backing_store, 2 + (col * 8), 2 + (row * 16), 8, 16, + col_bg_color); + PutS2D(widget->backing_store, Compositor.theme.font.monospace, 2 + (col * 8), + 2 + (row * 16), + widget->row[row + widget->scroll.y] + .col[col + widget->scroll.x] + .foreground, + , &char); +} + +U0 @umami_terminal_repaint_row(Window* win, TerminalWidget* widget, I64 row) +{ + I64 col; + for (col = 0; col < TERMINAL_MAX_COLS - 1; col++) + @umami_terminal_repaint_col(win, widget, row, col); +} + +U0 @umami_terminal_repaint_rows(Window* win, TerminalWidget* widget) +{ + I64 row; + for (row = 0; row < widget->size.rows - 1; row++) + @umami_terminal_repaint_row(win, widget, row); +} + +U0 @umami_terminal_ctrl_seq_cursor_move(Window* win, TerminalWidget* widget, + I64 dir) +{ + I64 pos = Str2I64(&widget->consumed_chars); + I64 prev_row = widget->cursor.y; + switch (dir) { + case 'A': + widget->cursor.y = Max(0, widget->cursor.y - pos); + break; + case 'B': + widget->cursor.y = Min(widget->cursor.y + pos, widget->size.rows - 1); + break; + case 'C': + widget->cursor.x = Min(widget->cursor.x + pos, widget->size.cols - 1); + break; + case 'D': + widget->cursor.x = Max(0, widget->cursor.x - pos); + break; + }; + @umami_terminal_repaint_row(win, widget, prev_row); +} + +U0 @umami_terminal_ctrl_seq_cursor_pos(Window* win, TerminalWidget* widget) +{ + I64 argc; + I64 i; + I64 val; + U8** argv = String.Split(&widget->consumed_chars, ';', &argc); + if (!argc) { + @umami_terminal_repaint_col(win, widget, widget->cursor.y, + widget->cursor.x); + widget->cursor.x = 0; + widget->cursor.y = 0; + return; + } + if (argc == 2) { + @umami_terminal_repaint_col(win, widget, widget->cursor.y, + widget->cursor.x); + widget->cursor.y = Str2I64(argv[0]) - 1; + widget->cursor.x = Str2I64(argv[1]) - 1; + widget->cursor.y = Min(widget->size.rows - 2, widget->cursor.y); + widget->cursor.x = Min(widget->size.cols - 2, widget->cursor.x); + widget->cursor.y = Max(0, widget->cursor.y); + widget->cursor.x = Max(0, widget->cursor.x); + //@umami_terminal_cleanup_strings(argc, argv); + return; + } +} + +U0 @umami_terminal_ctrl_seq_erase_screen(Window* win, TerminalWidget* widget) +{ + I64 i; + if (!StrLen(&widget->consumed_chars)) { + // TODO: clear from cursor to end of the screen + return; + } + I64 val = Str2I64(&widget->consumed_chars); + switch (val) { + case 0: + // TODO: clear from cursor to end of the screen + break; + case 1: + // TODO: clear from cursor to beginning of the screen + break; + case 2: + widget->scroll.x = 0; + widget->scroll.y = 0; + widget->max.x = 0; + widget->max.y = 0; + for (i = 0; i < 2000; i++) + @umami_terminal_set_row(win, widget, i, NULL); + @umami_terminal_repaint_rows(win, widget); + break; + default: + System.Log("Unrecognized attribute in ctrl_seq_erase_screen: %d", val); + break; + }; +} + +U0 @umami_terminal_ctrl_seq_erase_line(Window* win, TerminalWidget* widget) +{ + if (!StrLen(&widget->consumed_chars)) { + // TODO: clear from cursor to end of the line + return; + } + I64 val = Str2I64(&widget->consumed_chars); + switch (val) { + case 0: + // TODO: clear from cursor to end of the line + break; + case 1: + // TODO: clear from cursor to beginning of the line + break; + case 2: + @umami_terminal_set_row(win, widget, widget->cursor.y, ' '); + @umami_terminal_repaint_rows(win, widget); + break; + default: + System.Log("Unrecognized attribute in ctrl_seq_erase_line: %d", val); + break; + }; +} + +U0 @umami_terminal_send_response_string(Window* win, TerminalWidget* widget, + U8* string) +{ + if (!string || !widget->output) + return; + I64 i; + for (i = 0; i < StrLen(string); i++) + FifoU8Ins(widget->output, string[i]); +} + +U0 @umami_terminal_ctrl_seq_set_func(Window* win, TerminalWidget* widget, + Bool bool) +{ + if (!StrCmp(&widget->consumed_chars, "?25")) { + @umami_terminal_repaint_col(win, widget, widget->cursor.y, + widget->cursor.x); + widget->cursor.hidden = T(bool, FALSE, TRUE); + } +} + +U0 @umami_terminal_ctrl_seq_dev_attr_report(Window* win, + TerminalWidget* widget) +{ + if (!widget->output) + return; + if (StrFind("=", &widget->consumed_chars)) { + // Tertiary device attributes + @umami_terminal_send_response_string(win, widget, "\x1bP!|7E565445\x1b\\"); + return; + } + if (StrFind(">", &widget->consumed_chars)) { + // Secondary device attributes + @umami_terminal_send_response_string(win, widget, "\x1b[>65;6201;1c"); + return; + } + // Primary device attributes + @umami_terminal_send_response_string(win, widget, "\x1b[?65;1;9c"); +} + +U0 @umami_terminal_ctrl_seq_ecp_report(Window* win, TerminalWidget* widget) +{ + if (!widget->output) + return; + U8 resp_str[32]; + StrPrint(&resp_str, "\x1b[%d;%dR", widget->cursor.y + 1, + widget->cursor.x + 1); + if (!StrCmp(&widget->consumed_chars, "5")) + @umami_terminal_send_response_string(win, widget, "\x1b[0n"); + if (!StrCmp(&widget->consumed_chars, "6")) + @umami_terminal_send_response_string(win, widget, &resp_str); +} + +U0 @umami_terminal_ctrl_seq_req_tparm(Window* win, TerminalWidget* widget) +{ + // FIXME: allow unsolicited DECREPTPARMs + if (!StrCmp(&widget->consumed_chars, "1")) { + @umami_terminal_send_response_string(win, widget, + "\x1b[3;1;1;120;120;1;0x"); + return; + } + @umami_terminal_send_response_string(win, widget, "\x1b[2;1;1;120;120;1;0x"); +} + +U0 @umami_terminal_os_cmd(Window* win, TerminalWidget* widget) +{ + I64 argc; + I64 i; + I64 val; + U8** argv = String.Split(&widget->consumed_chars, ';', &argc); + if (!argc) + return; + val = Str2I64(argv[0]); + switch (val) { + case 0: + // Test set window title + StrPrint(win->title, "Terminal - %s", argv[1]); + break; + default: + break; + } + //@umami_terminal_cleanup_strings(argc, argv); +} + +U0 @umami_terminal_consume_char(Window* win, TerminalWidget* widget) +{ + U8 char[2]; + char[1] = NULL; + FifoU8Rem(widget->input, &char); + + switch (widget->state) { + case TERMINAL_STATE_CONSUME_BEGIN: + StrCpy(&widget->consumed_chars, ""); + switch (char[0]) { + case '7': // DEC Save Cursor + MemCpy(&widget->stored.attr, &widget->attr, + sizeof(@terminal_widget_attr)); + MemCpy(&widget->stored.color, &widget->color, + sizeof(@terminal_widget_color)); + MemCpy(&widget->stored.cursor, &widget->cursor, + sizeof(@terminal_widget_cursor)); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case '8': // Dec Restore Cursor + MemCpy(&widget->attr, &widget->stored.attr, + sizeof(@terminal_widget_attr)); + MemCpy(&widget->color, &widget->stored.color, + sizeof(@terminal_widget_color)); + MemCpy(&widget->cursor, &widget->stored.cursor, + sizeof(@terminal_widget_cursor)); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case '[': + widget->state = TERMINAL_STATE_CONSUME_CTRL_SEQ; + break; + case ']': + widget->state = TERMINAL_STATE_CONSUME_OS_CMD; + break; + default: + System.Log(Fs, "Undefined C1 control code: %c", char[0]); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + }; + break; + case TERMINAL_STATE_CONSUME_CTRL_SEQ: + switch (char[0]) { + case '0' ... '9': + case ';': + case '?': + case '>': + case '=': + String.Append(&widget->consumed_chars, &char); + break; + case 'A' ... 'D': + @umami_terminal_ctrl_seq_cursor_move(win, widget, char[0]); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'H': + @umami_terminal_ctrl_seq_cursor_pos(win, widget); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'J': + @umami_terminal_ctrl_seq_erase_screen(win, widget); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'K': + @umami_terminal_ctrl_seq_erase_line(win, widget); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'c': + @umami_terminal_ctrl_seq_dev_attr_report(win, widget); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'h': + @umami_terminal_ctrl_seq_set_func(win, widget, 1); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'l': + @umami_terminal_ctrl_seq_set_func(win, widget, 0); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'm': + @umami_terminal_ctrl_seq_sgr(win, widget); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'n': + @umami_terminal_ctrl_seq_ecp_report(win, widget); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 't': + widget->state = TERMINAL_STATE_CONSUME_END; + break; + case 'x': + @umami_terminal_ctrl_seq_req_tparm(win, widget); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + default: + System.Log(Fs, + "Invalid or unimplemented character in control sequence: '%c'", + char[0]); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + } + break; + case TERMINAL_STATE_CONSUME_OS_CMD: + switch (char[0]) { + case 7: + case '\\': + @umami_terminal_os_cmd(win, widget); + widget->state = TERMINAL_STATE_CONSUME_END; + break; + default: + String.Append(&widget->consumed_chars, &char); + break; + }; + break; + }; +} + +U0 @umami_terminal_output_char(Window* win, TerminalWidget* widget) +{ + if (widget->scroll.y != widget->max.y) { + widget->scroll.y = widget->max.y; + @umami_terminal_repaint_rows(win, widget); + } else { + widget->scroll.y = widget->max.y; + } + U8 char[2]; + char[1] = NULL; + FifoU8Rem(widget->input, &char); + switch (char[0]) { + case 27: + StrCpy(&widget->consumed_chars, ""); + widget->state = TERMINAL_STATE_CONSUME_BEGIN; + return; + break; + case 7: + // Audio.Beep(); + break; + case 8: + widget->row[widget->cursor.y + widget->scroll.y] + .col[widget->cursor.x + widget->scroll.x] + .char + = ' '; + widget->cursor.x--; + if (widget->cursor.x < 0) { + widget->cursor.y--; + widget->cursor.x = widget->size.cols - 2; + } + break; + case 10: + widget->cursor.y++; + widget->cursor.x = 0; + break; + case 13: + widget->cursor.x = 0; + break; + case 32...127: + widget->row[widget->cursor.y + widget->scroll.y] + .col[widget->cursor.x + widget->scroll.x] + .foreground + = widget->color.foreground; + widget->row[widget->cursor.y + widget->scroll.y] + .col[widget->cursor.x + widget->scroll.x] + .background + = widget->color.background; + widget->row[widget->cursor.y + widget->scroll.y] + .col[widget->cursor.x + widget->scroll.x] + .char + = char[0]; + widget->cursor.x++; + break; + default: + widget->row[widget->cursor.y + widget->scroll.y] + .col[widget->cursor.x + widget->scroll.x] + .foreground + = widget->color.foreground; + widget->row[widget->cursor.y + widget->scroll.y] + .col[widget->cursor.x + widget->scroll.x] + .background + = widget->color.background; + widget->row[widget->cursor.y + widget->scroll.y] + .col[widget->cursor.x + widget->scroll.x] + .char + = '?'; + widget->cursor.x++; + break; + }; + if (widget->cursor.x >= widget->size.cols - 1) { + widget->cursor.y++; + widget->cursor.x = 0; + } + if (widget->cursor.y >= widget->size.rows - 1) { + widget->cursor.y--; + widget->max.y++; + widget->scroll.y = widget->max.y; + @umami_terminal_set_row(win, widget, widget->cursor.y, NULL); + @umami_terminal_repaint_rows(win, widget); + } + @umami_terminal_set_col(win, widget, widget->cursor.y, widget->cursor.x, ' '); +} + +U0 @umami_terminal_repaint(Window* win, TerminalWidget* widget, I64 x, I64 y) +{ + + I64 current_col = NULL; + I64 current_row = NULL; + + @umami_terminal_calculate_size(win, widget); + + if (widget->refresh) { + @umami_terminal_repaint_rows(win, widget); + widget->refresh = FALSE; + } + + while (FifoU8Cnt(widget->input)) { + current_col = widget->cursor.x; + current_row = widget->cursor.y; + switch (widget->state) { + case TERMINAL_STATE_OUTPUT: + @umami_terminal_output_char(win, widget); + break; + case TERMINAL_STATE_CONSUME_BEGIN: + case TERMINAL_STATE_CONSUME_CTRL_SEQ: + case TERMINAL_STATE_CONSUME_OS_CMD: + @umami_terminal_consume_char(win, widget); + break; + } + if (widget->state == TERMINAL_STATE_OUTPUT) + @umami_terminal_repaint_col(win, widget, current_row, current_col); + if (widget->state == TERMINAL_STATE_CONSUME_END) + widget->state = TERMINAL_STATE_OUTPUT; + } + if (widget->state == TERMINAL_STATE_OUTPUT && !widget->cursor.hidden && widget->scroll.y == widget->max.y) + Rect2D(widget->backing_store, 2 + (widget->cursor.x * 8), + 2 + (widget->cursor.y * 16), 8, 16, widget->color.cursor); +} + +I64 @umami_input_get_text_width(BitmapFontTextInputWidget* widget) +{ + I64 x; + I64 y; + I64 text_width = PutS2D(NULL, widget->font, 0, 0, Color(0, 0, 0), -1, &widget->text); + + Context2D* ctx = NewContext2D(text_width, widget->font->line_height); + Fill2D(ctx, 0); + PutS2D(ctx, widget->font, 0, 0, Color(128, 255, 255), -1, &widget->text); + + for (x = ctx->width - 1; x > -1; x--) { + for (y = 0; y < ctx->height - 1; y++) { + if (ctx->fb[(ctx->width * y) + x]) { + DelContext2D(ctx); + return x + 2; + } + } + } + DelContext2D(ctx); + return x; +} + +U0 @umami_input_draw_cursor(Window* win, BitmapFontTextInputWidget* widget, + I64 index, I64 x, I64 y, I64 pos) +{ + if (widget->selected_region_start != -1 && widget->selected_region_end != -1) + return; + if (widget == win->focused_widget) { + if (Blink && index == widget->cursor_index) { + VLine2D(win->render_ctx, x + 2 + pos, y + 2, + y + 2 + widget->font->line_height, Color(0, 0, 0)); + } + } +} + +U0 @umami_input_repaint(Window* win, I64 event, + BitmapFontTextInputWidget* widget, I64 x, I64 y) +{ + + if (!widget->font) + widget->font = Compositor.theme.font.sans; + U8* text; + U8 ch[2]; + I64 index; + I64 text_width; + I64 text_x_pos; + ch[1] = NULL; + widget->height = widget->font->line_height + 6; + + HLine2D(win->render_ctx, x, y, x + widget->width, Color(156, 156, 156)); + VLine2D(win->render_ctx, x, y, y + widget->height, Color(156, 156, 156)); + HLine2D(win->render_ctx, x + 1, y + widget->height, x + widget->width, + Color(190, 190, 190)); + VLine2D(win->render_ctx, x + widget->width, y + 1, y + widget->height, + Color(190, 190, 190)); + Rect2D(win->render_ctx, x + 1, y + 1, widget->width - 1, widget->height - 1, + Color(255, 255, 255)); + + if (widget == win->focused_widget) { + HLine2D(win->render_ctx, x + 1, y + 1, x + widget->width - 1, + Color(0, 0, 0)); + VLine2D(win->render_ctx, x + 1, y + 1, y + widget->height - 2, + Color(0, 0, 0)); + HLine2D(win->render_ctx, x + 1, y + widget->height - 1, + x + widget->width - 1, Color(0, 0, 0)); + VLine2D(win->render_ctx, x + widget->width - 1, y + 1, + y + widget->height - 1, Color(0, 0, 0)); + } else { + widget->in_drag = FALSE; + } + + if (!Mouse.left) { + widget->in_drag = FALSE; + widget->mouse_drag_index = -1; + } + + Context2D* text_input_ctx = NewContext2D(widget->width - 4, widget->height); + + if (widget->is_password) { + StrCpy(widget->password, widget->text); + for (text_x_pos = 0; text_x_pos < StrLen(widget->password); text_x_pos++) + widget->password[text_x_pos] = '*'; + } + text = T(widget->is_password, &widget->password, &widget->text); + index = 0; + text_width; + text_x_pos = 0; + while (*text) { + ch[0] = *text; + text_width = PutS2D(NULL, widget->font, 0, 0, Color(0, 0, 0), -1, &ch) + 2; + + if (event == CPZ_MSG_WIN_LEFT_BTN_DOWN && !widget->in_drag && Mouse.x >= win->x + widget->x + -widget->x_offset + text_x_pos && Mouse.x <= win->x + widget->x + -widget->x_offset + text_x_pos + text_width) { + widget->cursor_index = index; + widget->mouse_drag_index = widget->cursor_index - 1; + widget->mouse_drag_origin_x = Mouse.x - win->x - widget->x + widget->x_offset; + } + + if (widget->in_drag) { + if (Mouse.x - win->x - widget->x < widget->mouse_drag_origin_x && Mouse.x >= win->x + widget->x + -widget->x_offset + text_x_pos && Mouse.x <= win->x + widget->x + -widget->x_offset + text_x_pos + text_width) { + widget->selected_region_start = index; + widget->selected_region_end = widget->mouse_drag_index; + } + if (Mouse.x - win->x - widget->x >= widget->mouse_drag_origin_x && Mouse.x >= win->x + widget->x + -widget->x_offset + text_x_pos + text_width) { + widget->selected_region_start = widget->mouse_drag_index + 1; + widget->selected_region_end = index; + } + } + + if (index >= widget->selected_region_start && index <= widget->selected_region_end) { + Rect2D(text_input_ctx, -widget->x_offset + text_x_pos, 0, text_width, + widget->font->line_height + 3, Compositor.theme.color.hilight); + } + if (index >= widget->selected_region_start && index <= widget->selected_region_end) + PutS2D(text_input_ctx, widget->font, -widget->x_offset + text_x_pos, 2, + Color(255, 255, 255), -1, &ch); + else + PutS2D(text_input_ctx, widget->font, -widget->x_offset + text_x_pos, 2, + widget->color, -1, &ch); + + @umami_input_draw_cursor(win, widget, index, x, y, + -widget->x_offset + text_x_pos); + text_x_pos += text_width; + + text++; + index++; + } + + if (event == CPZ_MSG_WIN_LEFT_BTN_DOWN && !widget->in_drag && Mouse.x > win->x + widget->x + -widget->x_offset + text_x_pos && widget == win->focused_widget) { + widget->cursor_index = StrLen(&widget->text); + widget->mouse_drag_index = widget->cursor_index; + widget->mouse_drag_origin_x = Mouse.x - win->x - widget->x; + } + + if (event == CPZ_MSG_WIN_LEFT_BTN_DOWN && widget == win->focused_widget) { + @widget_input_clear_selected_region(widget); + widget->in_drag = TRUE; + } + + if (-widget->x_offset + text_x_pos < text_input_ctx->width) + @umami_input_draw_cursor(win, widget, index, x, y, + -widget->x_offset + text_x_pos); + + Blot2D(win->render_ctx, x + 3, y + 2, text_input_ctx); + DelContext2D(text_input_ctx); +} + +U0 @umami_label_repaint(Window* win, BitmapFontTextLabelWidget* widget, I64 x, + I64 y) +{ + if (!widget->font) + widget->font = Compositor.theme.font.sans; + if (widget->text) + PutS2D(win->render_ctx, widget->font, x, y, widget->color, -1, + widget->text); +} + +U0 @umami_list_view_repaint(Window* win, ListViewWidget* widget, I64 x, I64 y) +{ +} + +U0 @umami_menu_item_repaint(Window* win, MenuItemWidget* widget, I64 x, I64 y) +{ + U32 text_color = Color(0, 0, 0); + I64 text_x = 8; + I64 text_y = 4 + (widget->height / 2) - (16 / 2); + I64 icon_x = 4; + I64 icon_y = 0; + if (widget->icon) { + icon_y = 4 + (widget->height / 2) - (widget->icon->height / 2) - 3; + } + if (@widget_is_hovered(win->x + x, win->y + y, widget)) { + Rect2D(win->render_ctx, x, y, widget->width, widget->height, + Compositor.theme.color.hilight); + text_color = Color(255, 255, 255); + if (widget->submenu) { + Gui.Window.Show(widget->submenu); + Gui.Window.Refresh(widget->submenu); + if (Compositor.GetWindowByZIndex(Compositor.max_z_index - 3) != widget->submenu) { + Gui.Window.SetZIndex(widget->submenu, Compositor.max_z_index - 3); + } + } + } + if (!(@widget_is_hovered(win->x + x, win->y + y, widget))) { + if (widget->submenu) { + Gui.Window.Hide(widget->submenu); + } + } + if (!widget->font) + widget->font = Compositor.theme.font.menu; + if (widget->icon) { + Blot2D(win->render_ctx, x + icon_x, y + icon_y, widget->icon); + text_x += 20; + } + if (widget->text) + PutS2D(win->render_ctx, widget->font, x + text_x, y + text_y, + T(widget->color, widget->color, text_color), -1, widget->text); +} + +U0 @umami_radio_repaint(Window* win, RadioButtonWidget* widget, I64 x, I64 y) +{ + if (widget == win->mouse_down_widget && + @widget_is_hovered(win->x + x, win->y + y, widget)) { + Blot2D(win->render_ctx, x, y, + T(widget->selected, umami_widget_radio_selected, + umami_widget_radio_active)); + return; + } + if (widget == win->mouse_down_widget->echo && widget != win->mouse_down_widget && !widget->selected && + @widget_is_hovered(win->x + x, win->y + y, win->mouse_down_widget)) { + Blot2D(win->render_ctx, x, y, + T(win->mouse_down_widget(RadioButtonWidget*)->selected, + umami_widget_radio_selected, umami_widget_radio_active)); + return; + } + Blot2D(win->render_ctx, x, y, + T(widget->selected, umami_widget_radio_selected, umami_widget_radio)); +} + +U0 @umami_radio_select(Window* win, RadioButtonWidget* widget) +{ + @window_widgets_list* widgets_list; + widgets_list = win->widget; + while (widgets_list) { + if (widgets_list->widget && widgets_list->widget->type == WIDGET_TYPE_RADIO) + if (widgets_list->widget(RadioButtonWidget*)->group == widget->group) + widgets_list->widget(RadioButtonWidget*)->selected = T(widgets_list->widget == widget, TRUE, FALSE); + widgets_list = widgets_list->next; + } +} + +Bool @umami_horz_slider_bar_is_hovered(I64 x, I64 y, Widget* widget) +{ + no_warn y; + if (Mouse.x > x - 7 && Mouse.x < x + widget->width + 7) + return TRUE; + return FALSE; +} + +Bool @umami_horz_slider_is_hovered(Window* win, HorizontalSliderWidget* widget, + I64 x, I64 y, I64 scroll_pos) +{ + if (@widget_is_hovered(win->x + x, win->y + y - 24, widget) && Mouse.x > -4 + (win->x + widget->x - 7) + scroll_pos && Mouse.x < (win->x + widget->x - 7) + scroll_pos + 16) + return TRUE; + return FALSE; +} + +U0 @umami_horz_slider_repaint(Window* win, I64 event, + HorizontalSliderWidget* widget, I64 x, I64 y) +{ + I64 scroll = widget->scroll; + widget->height = 12; + if (widget->in_drag && + @umami_horz_slider_bar_is_hovered(win->x + x, win->y + y - 24, widget)) { + if (widget->origin.mouse_x != Mouse.x) { + widget->scroll += (Mouse.x - widget->origin.mouse_x); + widget->origin.mouse_x = Mouse.x; + } + } + + Rect2D(win->render_ctx, x, y + 5, widget->width, 4, Color(156, 156, 156)); + if (@umami_horz_slider_is_hovered(win, widget, x, y, widget->scroll) && event == CPZ_MSG_WIN_LEFT_BTN_DOWN) { + win->mouse_down_widget = widget; + widget->origin.mouse_x = Mouse.x; + widget->in_drag = TRUE; + } else { + if (@widget_is_hovered(win->x + x, win->y + y, widget) && event == CPZ_MSG_WIN_LEFT_BTN_DOWN) { + widget->scroll = (Mouse.x - win->x - x + 2); + widget->origin.mouse_x = Mouse.x; + widget->in_drag = TRUE; + } + } + + if (@widget_is_hovered(win->x + x, win->y + y, widget) && event == CPZ_MSG_WIN_MOUSE_WHEEL) { + widget->scroll -= Compositor.mouse.z; + } + + widget->scroll = Max(0, widget->scroll); + widget->scroll = Min(widget->width - 1, widget->scroll); + widget->value = ToI64((widget->max * 1.0) / (widget->width * 1.0) * (widget->scroll * 1.0)); + + if (!Mouse.left) + widget->in_drag = FALSE; + + HLine2D(win->render_ctx, x, y + 1, x + widget->width, Color(190, 190, 190)); + HLine2D(win->render_ctx, x + 1, y + 4, x + widget->width, Color(0, 0, 0)); + VLine2D(win->render_ctx, x + 1, y + 4, y + 8, Color(0, 0, 0)); + VLine2D(win->render_ctx, x, y + 4, y + 9, Color(190, 190, 190)); + HLine2D(win->render_ctx, x, y + 9, x + widget->width, Color(255, 255, 255)); + VLine2D(win->render_ctx, x + widget->width, y + 4, y + 9, + Color(255, 255, 255)); + + VLine2D(win->render_ctx, x, y + 11, y + 16, Color(190, 190, 190)); + VLine2D(win->render_ctx, x + 1, y + 11, y + 16, Color(255, 255, 255)); + + VLine2D(win->render_ctx, x + (widget->width / 2) - 1, y + 11, y + 16, + Color(190, 190, 190)); + VLine2D(win->render_ctx, x + (widget->width / 2), y + 11, y + 16, + Color(255, 255, 255)); + + VLine2D(win->render_ctx, x + widget->width - 1, y + 11, y + 16, + Color(190, 190, 190)); + VLine2D(win->render_ctx, x + widget->width, y + 11, y + 16, + Color(255, 255, 255)); + + Blot2D(win->render_ctx, (x - 7) + widget->scroll, y, + umami_widget_horz_slider); + + if (widget->scroll != scroll) + widget->change = TRUE; +} + +Bool @umami_vert_slider_bar_is_hovered(I64 x, I64 y, Widget* widget) +{ + no_warn x; + if (Mouse.y > y - 7 && Mouse.y < y + widget->height + 7) + return TRUE; + return FALSE; +} + +Bool @umami_vert_slider_is_hovered(Window* win, VerticalSliderWidget* widget, + I64 x, I64 y, I64 scroll_pos) +{ + if (@widget_is_hovered(win->x + x - 24, win->y + y, widget) && Mouse.y > -4 + (win->y + widget->y - 7) + scroll_pos && Mouse.y < (win->y + widget->y - 7) + scroll_pos + 16) + return TRUE; + return FALSE; +} + +U0 @umami_vert_slider_repaint(Window* win, I64 event, + VerticalSliderWidget* widget, I64 x, I64 y) +{ + I64 scroll = widget->scroll; + widget->width = 12; + if (widget->in_drag && + @umami_vert_slider_bar_is_hovered(win->x + x, win->y + y, widget)) { + if (widget->origin.mouse_y != Mouse.y) { + widget->scroll += (Mouse.y - widget->origin.mouse_y); + widget->origin.mouse_y = Mouse.y; + } + } + Rect2D(win->render_ctx, x + 5, y, 4, widget->height, Color(156, 156, 156)); + if (@umami_horz_slider_is_hovered(win, widget, x, y, widget->scroll) && event == CPZ_MSG_WIN_LEFT_BTN_DOWN) { + win->mouse_down_widget = widget; + widget->origin.mouse_y = Mouse.y; + widget->in_drag = TRUE; + } else { + if (@widget_is_hovered(win->x + x, win->y + y, widget) && event == CPZ_MSG_WIN_LEFT_BTN_DOWN) { + widget->scroll = (Mouse.y - win->y - y + 1); + widget->origin.mouse_y = Mouse.y; + widget->in_drag = TRUE; + } + } + if (@widget_is_hovered(win->x + x, win->y + y, widget) && event == CPZ_MSG_WIN_MOUSE_WHEEL) { + widget->scroll += Compositor.mouse.z; + } + widget->scroll = Max(0, widget->scroll); + widget->scroll = Min(widget->height - 1, widget->scroll); + widget->value = ToI64((widget->max * 1.0) / (widget->height * 1.0) * (widget->scroll * 1.0)); + if (!Mouse.left) + widget->in_drag = FALSE; + VLine2D(win->render_ctx, x + 1, y, y + widget->height, Color(190, 190, 190)); + VLine2D(win->render_ctx, x + 4, y + 1, y + widget->height, Color(0, 0, 0)); + HLine2D(win->render_ctx, x + 4, y, x + 9, Color(0, 0, 0)); + VLine2D(win->render_ctx, x + 9, y, y + widget->height, Color(255, 255, 255)); + HLine2D(win->render_ctx, x + 4, y + widget->height, x + 9, + Color(255, 255, 255)); + HLine2D(win->render_ctx, x + 11, y, x + 16, Color(190, 190, 190)); + HLine2D(win->render_ctx, x + 11, y + 1, x + 16, Color(255, 255, 255)); + HLine2D(win->render_ctx, x + 11, y + (widget->height / 2) - 1, x + 16, + Color(190, 190, 190)); + HLine2D(win->render_ctx, x + 11, y + (widget->height / 2), x + 16, + Color(255, 255, 255)); + HLine2D(win->render_ctx, x + 11, y + widget->height - 1, x + 16, + Color(190, 190, 190)); + HLine2D(win->render_ctx, x + 11, y + widget->height, x + 16, + Color(255, 255, 255)); + Blot2D(win->render_ctx, x, (y - 7) + widget->scroll, + umami_widget_vert_slider); + + if (widget->scroll != scroll) + widget->change = TRUE; +} + +U0 @umami_vert_scroll_button_repaint(Window* win, I64 x, I64 y, I64 width, + I64 height) +{ + I64 i; + + Rect2D(win->render_ctx, x, y, width, height, Color(192, 192, 192)); + for (i = 0; i < height; i++) { + Line2D(win->render_ctx, x, y, x, y + height - 1, Color(192, 192, 192)); + Line2D(win->render_ctx, x + 1, y, x + 1, y + height - 1, + Color(255, 255, 255)); + Line2D(win->render_ctx, x + 14, y, x + width - 2, y + height - 1, + Color(128, 128, 128)); + Line2D(win->render_ctx, x + 15, y, x + width - 1, y + height - 1, + Color(64, 64, 64)); + } + + Plot2D(win->render_ctx, x + 1, y, Color(192, 192, 192)); + Plot2D(win->render_ctx, x + 14, y, Color(192, 192, 192)); + Plot2D(win->render_ctx, x + 15, y, Color(64, 64, 64)); + + Line2D(win->render_ctx, x + 1, y + 1, x + width - 2, y + 1, + Color(255, 255, 255)); + Plot2D(win->render_ctx, x + 14, y + 1, Color(128, 128, 128)); + Plot2D(win->render_ctx, x + 15, y + 1, Color(64, 64, 64)); + + Line2D(win->render_ctx, x + 1, y + height - 2, x + width - 2, y + height - 2, + Color(128, 128, 128)); + + Line2D(win->render_ctx, x, y + height - 1, x + width - 1, y + height - 1, + Color(64, 64, 64)); +} + +U0 @umami_vert_scrollbar_repaint(Window* win, I64 event, + VerticalScrollBarWidget* widget, I64 x, + I64 y) +{ + I64 scroll = widget->scroll; + I64 segment_length = widget->length; + I64 i, j; + Bool fill = TRUE; + + if (widget->in_drag && Mouse.y > win->y + widget->y && Mouse.y < win->y + widget->y + widget->height) { + if (widget->origin.mouse_y != Mouse.y) { + widget->scroll += (Mouse.y - widget->origin.mouse_y); + widget->origin.mouse_y = Mouse.y; + } + } + if (@widget_is_hovered(win->x + x, win->y + y, widget) && event == CPZ_MSG_WIN_LEFT_BTN_DOWN) { + if (Mouse.y > win->y + y + 16 && Mouse.y < win->y + y + widget->height - 16) { + widget->scroll = (Mouse.y - win->y - y - (widget->length / 2)); + widget->origin.mouse_y = Mouse.y; + widget->in_drag = TRUE; + } + } + if (@widget_is_hovered(win->x + x, win->y + y, widget) && event == CPZ_MSG_WIN_MOUSE_WHEEL) { + widget->scroll += Compositor.mouse.z; + } + widget->scroll = Max(16, widget->scroll); + widget->scroll = Min(widget->height - 1, widget->scroll); + while (widget->scroll + segment_length > widget->height - 16) + widget->scroll--; + widget->value = ToI64((widget->max * 1.0) / (widget->height * 1.0) * (widget->scroll * 1.0)); + if (!Mouse.left) + widget->in_drag = FALSE; + + Rect2D(win->render_ctx, x, y, widget->width, widget->height, + Color(127, 127, 127)); + + for (i = 0; i < widget->width; i++) { + for (j = 16; j < widget->height - 16; j++) { + if (fill) + Plot2D(win->render_ctx, i + x, j + y, Color(255, 255, 255)); + fill = !fill; + } + fill = !fill; + } + + @umami_vert_scroll_button_repaint(win, x, y, widget->width, 16); + @umami_vert_scroll_button_repaint(win, x, y + widget->height - 16, + widget->width, 16); + + Blot2D(win->render_ctx, x, y, umami_widget_sb_up_button); + Blot2D(win->render_ctx, x, y + widget->height - 16, + umami_widget_sb_down_button); + + @umami_vert_scroll_button_repaint(win, x, y + widget->scroll, widget->width, + segment_length); + + if (widget->scroll != scroll) + widget->change = TRUE; +} + +U0 @umami_widgets_repaint(Window* win, I64 event, I64 origin_x, I64 origin_y) +{ + @window_widgets_list* widgets_list; + Widget* widget; + Context2D* ctx; + Bool widget_is_hovered; + I64 x, y; + F64 ttf_point_to_size; + no_warn ctx; + no_warn ttf_point_to_size; + + if (event == CPZ_MSG_WIN_LEFT_BTN_DOWN) + win->focused_widget = NULL; + + widget_is_hovered = FALSE; + widgets_list = win->widget; + while (widgets_list) { + if (widgets_list->widget) { + widget = widgets_list->widget; + x = win->x + origin_x + widget->x; + y = win->y + origin_y + widget->y; + + if (@gui_window_is_hovered(win) && @widget_is_hovered(x, y, widget)) { + win->hovered_widget = widget; + widget_is_hovered = TRUE; + } + + if (@widget_is_hovered(x, y, widget) && event == CPZ_MSG_WIN_LEFT_BTN_DOWN) { + win->focused_widget = widget; + win->mouse_down_widget = widget; + break; + } + } + widgets_list = widgets_list->next; + } + + if (!widget_is_hovered) + win->hovered_widget = NULL; + + if (!Mouse.left && win->mouse_down_widget) { + widget = win->mouse_down_widget; + x = win->x + origin_x + widget->x; + y = win->y + origin_y + widget->y; + if (@widget_is_hovered(x, y, widget) && widget->callback.clicked) + widget->callback.clicked(widget); + if (@widget_is_hovered(x, y, widget) && widget->echo) { + if (widget->echo->callback.clicked) + widget->echo->callback.clicked(widget->echo); + } + if (@widget_is_hovered(x, y, widget)) { + switch (widget->type) { + case WIDGET_TYPE_CHECKBOX: + widget(CheckBoxWidget*)->checked = !widget(CheckBoxWidget*)->checked; + break; + case WIDGET_TYPE_RADIO: + @umami_radio_select(win, widget); + break; + default: + break; + } + } + if (@widget_is_hovered(x, y, widget) && widget->echo) { + switch (widget->echo->type) { + case WIDGET_TYPE_CHECKBOX: + widget->echo(CheckBoxWidget*)->checked = !widget->echo(CheckBoxWidget*)->checked; + break; + case WIDGET_TYPE_RADIO: + @umami_radio_select(win, widget->echo); + break; + default: + break; + } + } + win->mouse_down_widget = NULL; + } + + widgets_list = win->widget; + while (widgets_list) { + if (widgets_list->widget) { + widget = widgets_list->widget; + x = origin_x + widget->x; + y = origin_y + widget->y; + if (x > win->width - 12 || y > win->height - 12) + goto @umami_skip_widget_repaint; + switch (widget->type) { + case NULL: + break; + case WIDGET_TYPE_BUTTON: + @umami_button_repaint(win, widget, x, y); + break; + case WIDGET_TYPE_CHECKBOX: + @umami_checkbox_repaint(win, widget, x, y); + break; + case WIDGET_TYPE_TERMINAL: + @umami_terminal_repaint(win, widget, x, y); + if (widget->backing_store) + CopyRect2D(win->render_ctx, x, y, widget->backing_store); + break; + case WIDGET_TYPE_CONTEXT2D: + if (widget(Context2DWidget*)->ctx) + if (widget(Context2DWidget*)->fast_copy) + CopyRect2D(win->render_ctx, x, y, widget(Context2DWidget*)->ctx); + else + Blot2D(win->render_ctx, x, y, widget(Context2DWidget*)->ctx); + break; + case WIDGET_TYPE_RADIO: + @umami_radio_repaint(win, widget, x, y); + break; + case WIDGET_TYPE_INPUT: + @umami_input_repaint(win, event, widget, x, y); + break; + case WIDGET_TYPE_LABEL: + @umami_label_repaint(win, widget, x, y); + break; + case WIDGET_TYPE_LISTVIEW: + @umami_list_view_repaint(win, widget, x, y); + break; + case WIDGET_TYPE_MENU_ITEM: + @umami_menu_item_repaint(win, widget, x, y); + break; + case WIDGET_TYPE_HORZ_SLIDER: + @umami_horz_slider_repaint(win, event, widget, x, y); + break; + case WIDGET_TYPE_VERT_SLIDER: + @umami_vert_slider_repaint(win, event, widget, x, y); + break; + case WIDGET_TYPE_HORZ_SCROLLBAR: + //@umami_horz_scrollbar_repaint(win, event, widget, x, y); + break; + case WIDGET_TYPE_VERT_SCROLLBAR: + @umami_vert_scrollbar_repaint(win, event, widget, x, y); + break; + default: + break; + } + if (widget->change) { + if (widget->callback.change) + widget->callback.change(widget); + if (widget->echo) { + if (widget->echo->callback.change) + widget->echo->callback.change(widget); + } + widget->change = FALSE; + } + } + @umami_skip_widget_repaint : widgets_list = widgets_list->next; + } +} + +U0 @umami_window_repaint(Window* win, I64 event) +{ + if (!win) + return; + + win->repainting = TRUE; + + win->render_ctx->width = win->width; + win->render_ctx->height = win->height; + if (!@gui_window_flag_is_set(win, WIN_FLAGS_NOFILL)) + MemSetU32(win->render_ctx->fb, Color(204, 204, 204, win->opacity), + win->width * win->height); + + I64 title_bar_x; + I64 title_bar_y; + I64 title_bar_width; + I64 title_bar_height = 0; + I64 window_button_x; + I64 origin_title_width; + I64 pass; + Bool repaint; + U8 window_title[1024]; + + if (@gui_window_flag_is_set(win, WIN_FLAGS_TITLE_BAR)) { + title_bar_height = 17; + } + + I64 widget_origin_x = 4; + I64 widget_origin_y = 4; + if (title_bar_height) + widget_origin_y = 7 + title_bar_height; + + @umami_widgets_repaint(win, event, widget_origin_x, widget_origin_y); + + // Draw window margins + if (!@gui_window_flag_is_set(win, WIN_FLAGS_NOFILL)) { + Rect2D(win->render_ctx, 0, 0, win->width, 4, + Color(204, 204, 204, win->opacity)); + Rect2D(win->render_ctx, 0, win->height - 4, win->width, 4, + Color(204, 204, 204, win->opacity)); + Rect2D(win->render_ctx, 0, 0, 4, win->height, + Color(204, 204, 204, win->opacity)); + Rect2D(win->render_ctx, win->width - 5, 0, 4, win->height, + Color(204, 204, 204, win->opacity)); + } + + // Draw window border + Line2D(win->render_ctx, 0, 0, win->width - 1, 0, + Color(255, 255, 255, win->opacity)); + Line2D(win->render_ctx, 0, 0, 0, win->height - 2, + Color(255, 255, 255, win->opacity)); + Line2D(win->render_ctx, 1, win->height - 1, win->width - 1, win->height - 1, + Color(156, 156, 156, win->opacity)); + Line2D(win->render_ctx, win->width - 1, 1, win->width - 1, win->height - 1, + Color(156, 156, 156, win->opacity)); + + title_bar_width = 0; + + if (@gui_window_flag_is_set(win, WIN_FLAGS_TITLE_BAR)) { + title_bar_x = 4; + title_bar_y = 4; + title_bar_width = win->width - 8; + + if (@gui_window_flag_is_set(win, WIN_FLAGS_ICON)) { + if (win->icon) + Blot2D(win->render_ctx, 4, 4, win->icon); + else + Blot2D(win->render_ctx, 4, 4, umami_default_icon); + title_bar_x = 24; + title_bar_width -= 20; + } + + origin_title_width = title_bar_width; + + repaint = FALSE; + + for (pass = 0; pass < 2; pass++) { + title_bar_width = origin_title_width; + window_button_x = win->width - 20; + if (@gui_window_flag_is_set(win, WIN_FLAGS_CLOSE_BUTTON)) { + if (pass) { + if (win->mouse.x > window_button_x && win->mouse.x < window_button_x + 16 && win->mouse.y > 5 && win->mouse.y < 21 && win->button.close) { + CopyRect2D(umami_title_bar_button, -32, -1, + umami_min_max_close_pressed); + if (!Mouse.left && win->callback.close) + win->callback.close(win); + } else + CopyRect2D(umami_title_bar_button, -32, -1, umami_min_max_close); + CopyRect2D(win->render_ctx, window_button_x, 5, + umami_title_bar_button); + } + if (win->left_btn_down.x > window_button_x && win->left_btn_down.x < window_button_x + 16 && win->left_btn_down.y > 5 && win->left_btn_down.y < 21 && Mouse.left && !win->button.close) { + win->button.close = TRUE; + repaint = TRUE; + } + window_button_x -= 16; + title_bar_width -= 17; + } + if (@gui_window_flag_is_set(win, WIN_FLAGS_MAX_BUTTON)) { + if (pass) { + if (win->mouse.x > window_button_x && win->mouse.x < window_button_x + 16 && win->mouse.y > 5 && win->mouse.y < 21 && win->button.maximize) { + CopyRect2D(umami_title_bar_button, -16, -1, + umami_min_max_close_pressed); + if (!Mouse.left && win->callback.maximize) + win->callback.maximize(win); + } + + else + CopyRect2D(umami_title_bar_button, -16, -1, umami_min_max_close); + CopyRect2D(win->render_ctx, window_button_x, 5, + umami_title_bar_button); + } + if (win->left_btn_down.x > window_button_x && win->left_btn_down.x < window_button_x + 16 && win->left_btn_down.y > 5 && win->left_btn_down.y < 21 && Mouse.left && !win->button.maximize) { + win->button.maximize = TRUE; + repaint = TRUE; + } + window_button_x -= 16; + title_bar_width -= 17; + } + if (@gui_window_flag_is_set(win, WIN_FLAGS_MIN_BUTTON)) { + if (pass) { + if (win->mouse.x > window_button_x && win->mouse.x < window_button_x + 16 && win->mouse.y > 5 && win->mouse.y < 21 && win->button.minimize) { + CopyRect2D(umami_title_bar_button, 0, -1, + umami_min_max_close_pressed); + if (!Mouse.left && win->callback.minimize) + win->callback.minimize(win); + } else + CopyRect2D(umami_title_bar_button, 0, -1, umami_min_max_close); + CopyRect2D(win->render_ctx, window_button_x, 5, + umami_title_bar_button); + } + if (win->left_btn_down.x > window_button_x && win->left_btn_down.x < window_button_x + 16 && win->left_btn_down.y > 5 && win->left_btn_down.y < 21 && Mouse.left && !win->button.minimize) { + win->button.minimize = TRUE; + repaint = TRUE; + } + window_button_x -= 16; + title_bar_width -= 17; + } + } + + if (!Mouse.left) { + if (win->button.close) + repaint = TRUE; + if (win->button.maximize) + repaint = TRUE; + if (win->button.minimize) + repaint = TRUE; + win->button.close = FALSE; + win->button.maximize = FALSE; + win->button.minimize = FALSE; + } + + if (repaint) { + @umami_window_repaint(win, event); + return; + } + + if (title_bar_width) { + win->title_bar_x = title_bar_x; + win->title_bar_width = title_bar_width; + } + + StrCpy(&window_title, &win->title); + if (win == Compositor.active_win) { + HGradientRect2D(win->render_ctx, title_bar_x, title_bar_y, + title_bar_width, title_bar_height, + Color(10, 36, 106, win->opacity), + Color(165, 201, 239, win->opacity)); + // Color(85, 18, 132, win->opacity), + // Color(233, 15, 240, win->opacity)); + PutS2D(win->render_ctx, Compositor.theme.font.menu, title_bar_x + 4, + title_bar_y + 4, Color(255, 255, 255, win->opacity), + title_bar_width, &window_title); + } else { + Rect2D(win->render_ctx, title_bar_x, title_bar_y, title_bar_width, + title_bar_height, Color(235, 235, 235, win->opacity)); + PutS2D(win->render_ctx, Compositor.theme.font.menu, title_bar_x + 4, + title_bar_y + 4, Color(156, 156, 156, win->opacity), + title_bar_width, &window_title); + } + + Line2D(win->render_ctx, title_bar_x, title_bar_y, + title_bar_x + title_bar_width - 1, title_bar_y, + Color(156, 156, 156, win->opacity)); + Line2D(win->render_ctx, title_bar_x, title_bar_y, title_bar_x, + title_bar_y + title_bar_height - 2, + Color(156, 156, 156, win->opacity)); + + Line2D(win->render_ctx, title_bar_x + title_bar_width - 1, title_bar_y, + title_bar_x + title_bar_width - 1, + title_bar_y + title_bar_height - 1, + Color(255, 255, 255, win->opacity)); + Line2D(win->render_ctx, title_bar_x, title_bar_y + title_bar_height - 1, + title_bar_x + title_bar_width - 1, + title_bar_y + title_bar_height - 1, + Color(255, 255, 255, win->opacity)); + + if (@gui_window_flag_is_set(win, WIN_FLAGS_RESIZABLE)) + Blot2D(win->render_ctx, win->width - 16, win->height - 16, + umami_corner_resize); + } + + win->backing_store->width = win->render_ctx->width; + win->backing_store->height = win->render_ctx->height; + if (!@gui_window_flag_is_set(win, WIN_FLAGS_NOFILL)) + MemCpyU32(win->backing_store->fb, win->render_ctx->fb, + win->render_ctx->width * win->render_ctx->height); + if (@gui_window_flag_is_set(win, WIN_FLAGS_NOFILL)) + MemCpyU32(win->backing_store->fb, win->render_ctx->fb, (win->width) * 20); + + win->repainting = FALSE; +} + +Compositor.theme.color.active_border = Color(0, 0, 0); +Compositor.theme.color.hilight = Color(8, 33, 107); + +BitmapFont* @umami_theme_bitmapfont_001 = @bitmapfont_new_from_context2d( + Image.FileToContext2D("M:/Media/Themes/Umami/BitmapFont/menu.png"), + "Eight Bit Dragon", + "`1234567890-=[]\;',./" + "~!@#\d%^&*()_+{}|:\"<>?" + "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"); +@umami_theme_bitmapfont_001->line_height = 12; +BitmapFonts.Add(@umami_theme_bitmapfont_001); + +BitmapFont* @umami_theme_bitmapfont_002 = @bitmapfont_new_from_context2d( + Image.FileToContext2D("M:/Media/Themes/Umami/BitmapFont/sans.png"), + "Nineteen Ninety Three", + "`1234567890-=[]\;',./" + "~!@#\d%^&*()_+{}|:\"<>?" + "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"); +@umami_theme_bitmapfont_002->line_height = 10; +BitmapFonts.Add(@umami_theme_bitmapfont_002); + +BitmapFont* @umami_theme_bitmapfont_003 = @bitmapfont_new_from_context2d( + Image.FileToContext2D("M:/Media/Themes/Umami/BitmapFont/monospace.png"), + "Spleen", + " !\"#\d%&'()*+,-./" + "0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`" + "abcdefghijklmnopqrstuvwxyz{|}~", + 7); +@umami_theme_bitmapfont_003->line_height = 16; +BitmapFonts.Add(@umami_theme_bitmapfont_003); + +Compositor.theme.font.menu = BitmapFonts.GetByName("Eight Bit Dragon"); +Compositor.theme.font.sans = BitmapFonts.GetByName("Nineteen Ninety Three"); +Compositor.theme.font.monospace = BitmapFonts.GetByName("Spleen"); + +Compositor.theme.window.min_width = 128; +Compositor.theme.window.min_height = 128; + +Compositor.theme.window_repaint = &@umami_window_repaint; + +Context2D** umami_pointer_wait_frames = CAlloc(sizeof(Context2D*) * 18); +Context2D* umami_pointer_wait_anim = Image.FileToContext2D("M:/Media/Themes/Umami/Pointer/wait.png"); + +U0 @umami_pointer_wait_anim_init() +{ + I64 i; + for (i = 0; i < 18; i++) { + umami_pointer_wait_frames[i] = NewContext2D(24, 24); + Blot2D(umami_pointer_wait_frames[i], 0, -(24 * i), umami_pointer_wait_anim); + } + DelContext2D(umami_pointer_wait_anim); +} + +@umami_pointer_wait_anim_init; +Compositor.theme.pointer.wait = Animation2D.NewFromFrames(umami_pointer_wait_frames, 18, 50); \ No newline at end of file diff --git a/Media/Themes/Umami/Widget/checkbox.png b/Media/Themes/Umami/Widget/checkbox.png new file mode 100644 index 0000000000000000000000000000000000000000..39eed00579691bce6c22cdf04ccc4d125cc6d401 GIT binary patch literal 233 zcmeAS@N?(olHy`uVBq!ia0y~yU{C;I4mJh`hT^KKFANL}Y)RhkE)4%caKYZ?lNlHo zI14-?iy0WWg+Z8+Vb&Z81_lQ95>H=O_Q!0DEK-USwh9X{Ffd4#xJHyX=jZ08=9Mrw z7o{eaq^2m8XO?6rxO@5rgg5euGcYg&db&7zv5SRu-3m%d eN=o&gn3noD-mT2F=wV=BVDNPHb6Mw<&;$T~8A$^G literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Widget/checkbox_active.png b/Media/Themes/Umami/Widget/checkbox_active.png new file mode 100644 index 0000000000000000000000000000000000000000..57eedc0eb2ec327a9fefaf31c5dae6bbae7efeb0 GIT binary patch literal 231 zcmeAS@N?(olHy`uVBq!ia0y~yU{C;I7G?$phQ^Te;|vT8oCO|{#S9GG!XV7ZFl&wk z0|NtliKnkC`(rjn7J0SeWkuf@7#Ji=Tq8=H^K)}k^GX<;i&7IyQd1PlGfOfQ+&z5* z!W;R-85kIZ0(?ST85kH65)yXo*m2{=jTbLo{Qv*ovi9{(1_lPkk|4ie28U-i(mS|xv6<249-QVi6yBi3gww484B*6z5(Hl zeBulY459%(A+8KykdTnDW5Vwgjdu>RDj1tltWZR#8s3- xgy-mbro%iu(FS)OGBJ5DrDZBKs2Ff*urgd?7ME5vSls|}fv2mV%Q~loCIB1cLMZ?M literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Widget/horz_slider.png b/Media/Themes/Umami/Widget/horz_slider.png new file mode 100644 index 0000000000000000000000000000000000000000..9d3fb382eb3144b6c80ed8c28292aeb0e0617f28 GIT binary patch literal 224 zcmeAS@N?(olHy`uVBq!ia0y~yU{C;I7G?$phQ^Te;|vT8oCO|{#S9GG!XV7ZFl&wk z0|NtliKnkC`(rjSF+r7!?rJ#<3=Gl%J|V6QU{Fy}F=4`l6DLkwzkdDw`}ZF{eE846 z;3IxUl!1YPu_VYZn8D%MjWiG^$=lt9qwU_*Dh37yJ5LwK5RLQ696?PCY!U(b?F@`2 zjY*0ojOikb#U_mk~$XUsseLC%?$= P2D!k~)z4*}Q$iB}SnxkE literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Widget/radio.png b/Media/Themes/Umami/Widget/radio.png new file mode 100644 index 0000000000000000000000000000000000000000..15963bf1ee5f601aab547d0178f6437efd9c05a2 GIT binary patch literal 255 zcmeAS@N?(olHy`uVBq!ia0y~yU{C;I7G?$phQ^Te;|vT8oCO|{#S9GG!XV7ZFl&wk z0|NtliKnkC`(rjn7JUWR=7$Ch3=EPbt`Q~9`MJ5Nc_j?aMX8A;sVNHOnI#zt?w-B@ z;f;La3=9nX0X`wF3=9kj2?;xP?6`5`#{d8Sm%mx1%)r3FSQ6wH%;50sMjD8dH_!@hljQC0!Y-EB@tT2wA=1;uF+^ixazX+JFK>ba+rneVmYq;qbF8W9&`S>{oi;D8 zrV~sDonCo$C9$4ha(%(%HG`>1gQ>}@OM_`mTT_$9whhZTxw;g&4W=w#x>Vd?LC2KE fpWGQ+85l&R1w{XU>H5yVz`)??>gTe~DWM4fd?{Ac literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Widget/sb_down_button.png b/Media/Themes/Umami/Widget/sb_down_button.png new file mode 100644 index 0000000000000000000000000000000000000000..0a80796323e775af4587a975a93612acdf541e5c GIT binary patch literal 115 zcmeAS@N?(olHy`uVBq!ia0y~yU{C;IMrH;E2G1=owlgp=um$*pxH5pj()69@85kHC zOM?7@862M7NMm4NkoI(O4B@z*oN$2qhrEG-!CwZ!dOn7oJv?977<719I={XaJPXq7 M>FVdQ&MBb@0Eqz|1^@s6 literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Widget/sb_left_button.png b/Media/Themes/Umami/Widget/sb_left_button.png new file mode 100644 index 0000000000000000000000000000000000000000..dadd1627fbb29f05207272722a5dc9646944fe59 GIT binary patch literal 116 zcmeAS@N?(olHy`uVBq!ia0y~yU{C;IMrH;E2G1=owlgp=um$*pxH5pj()69@85kHC zOM?7@862M7NMm4NknwbJ4B@z*oN$0ms$qtpi$OC-nFY5F+q9+P3=G?fnJ2`zZuSQ0 N_H^}gS?83{1OS7i91Z{g literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Widget/sb_right_button.png b/Media/Themes/Umami/Widget/sb_right_button.png new file mode 100644 index 0000000000000000000000000000000000000000..8d328a3f7aa06aa189fbc8f031a96fc83cc493c1 GIT binary patch literal 114 zcmeAS@N?(olHy`uVBq!ia0y~yU{C;IMrH;E2G1=owlgp=um$*pxH5pj()69@85kHC zOM?7@862M7NMm4Nkn(hK4B?1QPB_3c!-2(kMh1f*`vaXFI(r!y9*QttNiASj1nKp3 L^>bP0l+XkKLMs~v literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Widget/sb_up_button.png b/Media/Themes/Umami/Widget/sb_up_button.png new file mode 100644 index 0000000000000000000000000000000000000000..49e70bed677787d97ca38dde20e3bcd8d477acb7 GIT binary patch literal 115 zcmeAS@N?(olHy`uVBq!ia0y~yU{C;IMrH;E2G1=owlgp=um$*pxH5pj()69@85kHC zOM?7@862M7NMm4NkoI(O4B@z*oN$0!qH#v&N=A|Z`TRp2BdpgY5I)xx|@D|S!cyzhu1qu@@~DD^IrGh>JxnL#px)rq;aH7UY{d+;?dMC+%_(znj7EhS(Hs-U|?Wayp!SGmgf)r?i^hP O;(EIJxvX zaB^>EX>4U6ba`-PAZ2)IW&i+q+U=HIcH}4wh2L4lECC6G#B%sOXLiubp9^+5Rh7;p zIi2)V%(0<_EXnu7xZL>l?{xp*V9(J-QA;@`k0a)oxnN-aIP$)dkNdh0KBw^e!S1dJ zhAE&A$4@P%e}P?IZn)*3`e=7{&aj<<$UmeGEAtxn`yyfI!%p!Gl*chh+4R2DdIoBz z<1Xbq<5kwtt>5m!pirp`5(VZFDENJ~qk?dp1!vkW)}9v<)3dw(y` zgUGj$dshD%J&Didw1+1x)2R^#znvg?vWF`T-zN@NMf^Ack<+WkInOz}%{h*zD=|UQ zbgrl^Hr>L&u|o2&Z%b&3Q!v)uGMf3~8<2P3d_`-Ni9uyLiqxslq_O)(Z7p~n3?|In zfMwYXj50TXp%7Q2$r^I#!9cF@a3R3VS1q*j=AEyRA#*3VG9eh@mN%RGzsb)uH)mTC zk+(}L)WvIEql`svUEu{FG*51_1-`WtzrR~QSgIf>TV}%p7GIALRp@1Hh0Qs@PtYH) z5IOB<07!(~f?_CufKL>J5_x4-BIsxUR8TpG$Q1-gggc7l7(!yFY`k*g9;NNe!ikp| zdlNti-Yf;19138?C|Exl4Qr@KkSHQaRLn`1l0u3}Qc9ZECRa3QRMDiWrdf*xizb## z%`98VoCEei$ti2LoO3Dc3KA8>D)29`RIZ`MDm7KDR&y;4%csRAEj4Y{ax0yC=-4KD z>e{X6UIw{=onpixBMluk@+cE)ZH5`A%rte{%(Gn7PO7ikC#ca$jW?;~rx!I?o$4(F zt?fh>GZ5oQAg+r57Bnwr-Y7<191by~!LFb-mwbg_FO_aSah>_go6 zJ#ucL`wDUn=w5UC4Yj_$b8Qnl-omA+4|w}9Kdd;mvOa3%Py6o3OYyR0SVD(r7-RL2C!GR;3YT++NB&izRp_ek6mlS`&u>z3t^biuYUb{{;F8^b_bO&^HB|`16w8f3md6{_Q22!vFvQ zglR)VP)S2WAaHVTW@&6?001bFeUUv#!$2IxUsJUrEe;lx;*g;_Sr8R*)G8FALZ}s5 zbuhW~3z{?}EiR6NYr(;f#j1mgv#t)Vf*|+-;^gS0=prTlFDbN$@!+^0@9sVB-T^|R z%rvWO9MJTtnM%aPOmt?1%Ws?u4huXpY^GE5#9?Bw(7{Rvv!baH zPZ38|O{aVz*5I zM;yOIE`?krFmf!Q0u8d`2mgcL-CFs{2{$Pi2fANu`(p$M>;lcYZGRuzcJl=AKLb~K z+h1t_GoPf_+gkJp=-&n|uG^Zj2VCv|Lr;cm%8ulxDdh9O`x$*x78tk%`qtduTKhPC z0MgV|@&-6K1V#&#z3%bu-p=0sJ=5y%2PLR-hzx1`mH+?%24YJ`L;(K){{a7>y{D4^ z000SaNLh0L01FcU01FcV0GgZ_00007bV*G`2jm6_0W$>oG38|d002fwL_t(o!|l|; z2>>t%1VQ_+!Co)ld<1oxd!QK#++FQQNYdWyZ4=ztcT5mv|2jdGeeMLA?9V62 gWZ(CQ0002SJHo3(7r^SZSO5S307*qoM6N<$g1!utoB#j- literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Window/min_max_close.png b/Media/Themes/Umami/Window/min_max_close.png new file mode 100644 index 0000000000000000000000000000000000000000..d03a45ded519536760e8ff9f052d035befd53489 GIT binary patch literal 650 zcmV;50(Jd~P)EX>4Tx04R}tkv&MmKpe$iQ?(*34i=Q+kfAzR5EXIMDionYs1;guFuC*#nlvOW zE{=k0!NHHks)LKOt`4q(Aou~|}?mh0_0Yan9G^=YI(DbUA zO2oxXc2x|#q6YyCB7%_2EMrcRlJFc~_we!cF2=LG&;2<lN)#OK6gCS8#Dk?V@fZ=4Gb3p_Jyrc?98VPdh+!Ab|SqNx#25l2-`r+gvf zvdVdjvsSLL);;+PgE@U=nd>x%k-#FBAVGwJ8cHamiWsdrDHc++ANTM_9KS>^gt2Nx|9?rV_g0000OP)t-s0000$KtTWh{{R30KtMo%fPl=*%>V!Y?UJs%00003 zbW%=J0002`cEsKQ0004EOGiWp7Ga3%00014Nkl#_KWHLx1*(})%t3(=5BX$X(2GVrE?Jx$bKuEw{1(U$#4|JcG khJrx|I?dpM12EtM0IJJ>mjA`NO8@`>07*qoM6N<$g5u{8&Hw-a literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/Window/min_max_close_pressed.png b/Media/Themes/Umami/Window/min_max_close_pressed.png new file mode 100644 index 0000000000000000000000000000000000000000..f7a9c9a445b73d123d9efbe54c680fe64f6ec09e GIT binary patch literal 359 zcmeAS@N?(olHy`uVBq!ia0y~yU@%}{U=UzoW?*1g)bOy0fq{V~-O<;Pfnj4m_n$;o z1_lO&WRDrPK92^=N8qSq>Lsz PfjsT$>gTe~DWM4fk=|jK literal 0 HcmV?d00001 diff --git a/Media/Themes/Umami/wallpaper.jpg b/Media/Themes/Umami/wallpaper.jpg new file mode 100644 index 0000000000000000000000000000000000000000..1d7990c436a0fea3545395e9c6b1aac90f44ef87 GIT binary patch literal 527141 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 z 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

C0VRjf&^5` z+o}oNoP;Zi!j(Be-0^|sf!dCJQ$O3=^ew2&_PrfAZzgB_U%LKwjOQ@^71)STMZ((@ z5LbF5rz%IPa5ALph*t@Nr#+3iW2LduvTXSTWED8_YE#X>!ojMp0~B4fRD&rjGrk!V z05s+}%R2gM$hpGO;#2ksLBe!~SH*214+jn$;@JX&bLf zO;q@in}PwtR~ZU;@8kc6gA<)BnyMQuT{zYTB#^6A%A^;Ht4!b14AkYy9J2EBUwhl$ zQl=sX8-NwOgYji9jbxp&dyRPg)rPSvzW=U!J19JZkrFN@9^DHFQI}LULCf1Vdj-fC zoeY8;DK>BOLU^FZ>Bq@9hEwuPZ^$bGNQ;es(n9re;`Ok6WZJ2NhTSD~uyhD{)ldjz z*CS=%%m|~z!_QRuWq?S}$11btP~x)a`(i+!@Q7aUKc$)ek)*I09=7HrHrAZ;S&P$e z&40OsT3A0E6jM$%yvRE5#c#lOJ!Kr%Y|UtduD1xQt4~vS8z%do5Pyxbq_=qV)W~sg zz-VsU9{AfwPDV2YEHMRynie`bES}z%d^&a(NzAQj@5DhN%5eJTY4^~`(oLJaTp4V6 zu{43w4_=gJ4rOP?LVa=&JjI>6-=NTyi$f_MF*G+jc*Gc%xkud1EJx`E-6TG!X4f_hZx2k`KI|*%__ca;WhfH+85sx67EpLKx`dMkVW0VDB3Ak%+(gGxZFuuo zJBpq(WV7b$7QYF88D}ec%v?|$_v{@faYlFX-GHUY1V!b3)rCAAS;FS9Zt}#}mNZCF zjJ5@d<9<<O-+zYBuw^N!^vn)2ju^aQXq)p6}+7oHnfTm4NUdvO|Tr7i8 zW`^8~Bm~9^c$7C%T!WUsJP`$_y9YIC=49nNyUYoW9_ERpb8U9RWuykIc|ZlfXLE}t z5UrI74*;ILn~Kuiv$2W3!_MOJGxscYYlmsMx20q_7E+3!u!Hpl8+}A;wz*--X(tYg z%77h@nb~PtXFUKoyD9K0y)6k)Na!0ooOoE2$XZ+fikiDNUkMqR`ZRofDLvu+ z*8T&^OHh)vJF-%Bz}#=b24s`pE7ZbPgd%z&1dG!n)FUU%oJ>cu;`&I{)mHA`*C!fw zTXfQ}y;Y7bPXS(AS(EUCIYlEM0KB+f7>93B?qE~Jn+6T?-7n?IC+~zKa*nJhGj2K3 zHt?L7czK!v{w&f}Dy;7^X5=R6ffO|q6Z4wCh&PX_*!dGez5;EoW6KAQ9ln& zohkgMF;APB0W}qcZ;9iQc+F#;Lo`)}@e$z;ep~-i_HMJ3ONjDWlF)p8ilqZMbMmEF z@i{dCbJj3!3qCggVeH2(*N4vhZCir1jomt%v6`F?mESEC1;+nnxiV{Yl%@fajKiv`9MPD9L3O{GY)1ZK{ zr~SC&dH{Hz6=y=j#tA_c%*Dc0@h({T`>j0>{)`7e9+~WG2i+f^UOoVPd>68yVdDhD z`;;_*Rs;YGP)7eg@i#s5)z+)#y@6tw)Rl7*JUG z^Cw}~LZWJPvGIq-e0+bP*ayJue-}Ug9cLTAVu?mIe!c*GEzW-cjD%Y&=F5pzzmzA* z!F~?<56%-UL1|c%Zk7*-Cl%k1&qY4~TuN>LB@Vj%U!W8P9 z#UE^PcGhXziCAJ~tfsbpc8d9-NlILk^M5OUSa5Ysmj>-m)#W`k$Fmqj`z~8&l!Iq} z(J4u6hYWBXlb|M+X3xmBZD?>Fba&X=&%Q1)-8{tFd^|6;h^pQGvwco_C6p=oFP%ZYV=l&d63zaLBSneu9_ z`2(QJ^Z_uOG>xBUm6O{1E$maA@*AjfVJ81Zs+$8fJTaWe(qXG`SA z&YC2jYmT%<{U48*evb%B4ktf2-148mxCg+#u4~%sDGDHN^dI1iCiQ48bbqQAf6ra2 z&LPP{$2Aiup=O2BpcQ={wQ85FY}P59<-3W~y|rPoPa|8hjp_<20r#LF5GO9X?|+qh zti=hDtdeKxR!KEh3F3F_e@^0C&=~D=+OP~O7s~D)lZ)-k67e6qBpeI|O%!A2tC}m- z+0S~^DVYY>Ga^YEqjmbx#((#1wOqF+i|+=NWz>B~bgDm7!%9Mn3Z)34K@uNB!jf|D;h?{hPYvnozhYU!6zo|-%& z$={h4lC<#=`Y}@5LpYVr*x?7j^q9Mz!}?#j9eJSbd|LLiq_jW_IQx!#X}4^mSjO1h z^Ci>(dW+h4AY(90T}w=G%LrttSsIvaFAKBmZ_c|W_Usqd6y0D{m&6a#+vG?xN!K+z zC%=%k5mfq`kQW2Iw%SbrIde~t@PA6MPC@+Pp1{8iR{`dU8|Dn~)@_<4QKKgAxZu)2 z-mQkR;<|6k<ZhRd+QxSSYoE{yza9X;3OW+U>vj*)Z7+w!|8{(RNBT&0@tNy3 zaJ9a&D8!ag_C`sqlBCY%%d^Pk4|8T5k(OZ&`o|rCDsN*yeCmPq01v+%M)s*Prh1zAlC+gwL4%2Zb4<%r3QHcH7S^NNBpbBd3|1v4b?(mbK_Do#EKp#z8ubgfqs z&8V5DU~Ip`W{BY6^6WH&WrREcD%{lr3E_&bfNh`12d5a{p-_!(W3#_rK5odTpYcED z!OC67oDVKKUwSgvpFCrI}M~ z)hz$U28~brJX__K;=k5-I>=0K+Nxr`>hrf15B3)RIsH9WYki-^--AMO^c0%PHhcKN=ub){fBEG{_A4aALli!ZyaA>#lV z=}RL|F$pHhj=d9APD}GTiz>{hy1MKrAM+`;5F3Jw5eEJF@7cVnXXmW7T6&u!&JiEX zP8y;4<$~pIMK5sl!c@mA+L|p4`ox(T+lTYiF6?!dZ^dt%ET;&~hhl)A2^1?d3zG`S zBQ=B4gnG6eb=w(8&60EMFvFjaxIDA*D@PlIm6lU7Ll?frD@LyJ8T1;7o$s-^G(4)aa)VS>QuJ*;F@)c<&K>^Jw!7r3 z0MMlV#^zN}d0S3)0-%F-iENsx1@YtDZ)fNXH$$+HTZsk!`mI!T$Bhd-+Ud2O>=vJs z2Sm1U$IU@AaiO|>mpm~2e!ou$s&j1L;OLwVjB^A3xs!$3rer6Q?{@M6=C16t?O=lK zU=G*+wvw$sntQXPo{`FgE%|t|p|nsytd0|mkZM8|;7R^di#w^ z-MaL%NXkD_N)C*nQi`bMvevH=Y~^#oAep}E<=O71j}zm0%#%QutM$)DIKgk0D+hEU z42LNbJYKx5k?^C-fG5j>-#JdDCRxzcA9Wq2U5&B|bFW;gOMw+8 zuEga^YeobEg8ecBv{nipF(f%ir6vu*Ne$Jf0qC(2z=9}Yk0WL0%;yc6O3RM`mBIiP zMI3tb^f5EJ)h2bb+~@$Fjdad-cS1xO6k?6sV;f>@b&lJUIf-h@mZ~mQHE%^~;R)pJ z{?$=LGc$m|q{Y5V4zZx!*IcU}{%B*OIhx7(<8rM|CAFNZYOFPFmQB89-=8sUCQheo zr%x|jgIaO|-sY8ze{Oh-O<6IPdxpgp9cPHCkJ&ISms!$zy5t_Ru94Ht+8`Wbk+q9$ zceA=L#TXBszi9ESieIfn%^!{x@NqT$$}jyR#yGDbkf3}>fKJKh`QoY{m-v{@-J`$oeDHpbhz+m#&SG{`wp?y{MaY~ zE{x)0buO`n7R15%3R`FA;~VwOr%+H@>dRS+bKu9F-^Q$$2%}B22D7kqGeZCJ*9pV9 zTvQrvkBXm`$}1mBN?=6lu5>?Qt8aK(4X&Q85SqZHRz`)gTtbWbOv~9I5#9aWt!y!y zSCY3b*B3(1gOLZ)*12;}jAr~$VR$2)cHV*ga#J>X3?0lWz9|;8AFiNt*LkQ}{)ow4 zsaR)(eslF>Zf{C*e-)6KfUlQV+pntk}RC^HPWoG zr$atF!)h_a4l%T~-oC*KX3XzOzmzB}H7%sI&RCx}>{t_7FKCC&N#gr`#q|p1kReBN zacRBl)?P9b9A7)+Wm91Ozh-uxdtdwT`Vj|?&@Cj{(D_bNh?Dvd5VDFh;_+gZ1Cx+? z`0EtW*?6Y!QM?}%o<=B0J$}Re0q;$eo4Rs|R2_$-#loW+3&11VT)9z7Ufb?}+WVq= z#Ap@M&4@OH???t=@LEmeX_ zMz-koc4xJ9i~dM9j%#u~eZi_NFPGnbXVfWI*t@Ao<~@4H(B79~Np6}ba7$-^%k&C8 zbDLWW#H9baFnLEVdQSoSU>}>G7$Svh-0&Vwhl@@H6I5=9SQfM~;A39jyLbN4YD{=e z=T*20UH36t#MeKCy{}in>ko1lEPq9}FYOS#QH7pfT+D%?2@~UX{dhhIEbW@eh4?8CCL^}T`$#-@|rC<+MbzD)S+}& zQLNU&aR41>iPMd{{k{)Xa?w>UisS!mk-=iaviLf*^;JJhmVwT_e|30dt;9c95*0H? z&VVhX=BI_njk8a+^3qD(5oB)#Y5d#^Ltj6uC|FT-qbB#}(VeZeMGgp$B(`T&m%{@q z)$HuF`(Gyy^;P2Yo%8p%#@|+3hG|Uc+)L!yXYZnTGh+PM)D$B!-YJbPb!#+PA{+LB zItdESA@y$e1J4Y2lW1&bOfPW$ga& z$K=NmMTw=+)X(o=EWvVz{o){VsyFw7yL z%vk9Ga0{L)R98DDmuC54O{DLYW8z1!)AMWM(F0&LBf`17e56G6g0Y(PH|ZYY)}&N$ zYLEn!zC?TkGN@-XV7K$mNC*9D)uj#L7}t|8&Cl)&7BByBrov*3ra|00Wx?ebzQzE8v0Qp+q)X4F|r;Q06BJzrCX? z*eILbCSjl+CC}X6>?;+W&0_*#!>c0qFebIpf05%MP(F=T(C?{lS#eDX3M+ET%~Ogj zTu<`Jx?9F9=y5M*_+?4L@dDF5I<}j%hicWgEk;dcG2_`i@n$9|B|i?lj~cx<4QYD7 zIXEDxeamCR>#Hr@v;OLlA9179qwaw%6?+@6Dk&T-Lq^0QGx`P%RJAJZ2I$)trRt>i1l#t=5Q2VJg%H_4V2Q8!f!sx)iU)vSpcGc-@?W%g_H-QMuY8wM z)UgKp#}BhT12CCoyUo6-+Tz8q;~VcJo^wZsgTGE>Qdm`!oN0w|K5JqvAFk?g+51|} z&;TwpX?{Kc1X}w$9Uf`}p(GVTj?GhZbzN~H<=@69=>rbn) z^;70T8T$Z;wCKR_53g)s6vkQICGg(pEI9wO#?CC|{UT z$&mv_Hw@V5?v(B>0hN*xK|)mGT)!9RT<3qdUwvMDZ)1T5>PTTi8TATb?K@AKkn|9Q>wZB#%t=8-ixDL)8g_@b z;3#iv$CVsKs+Wk0NQX||CE=v;?O)>_;gmfOEAjeZ12Gf^pgZHnL@_yRE;Ncf)MBc- zs>3WZ#pDu{7K*#LLlr);Thgg}A*%BKH8B73{%7+SFyQjO{YEg$PG`t2Vj1SO3^3Dp zbLLwd@6lQ`gZTJ~w- zEiSJBv`%cTFaVyUVvQ}}ZU}Ag2GW$Y^CnXKJ|P6-zoRh9OGdZMMd>2yQvDx5aViSzAukuk4jmq9yWwiwg25=Sjk9W#IKq-dFiCLu^G9uxNF*b6cjOVL z0!(ZN8rpd+A)az{K~CDhaV2?xhzbc2AL^*L*(p?z;(Yeqc#cGrwT_0?_4U+;8Wh~D zszQKR4xD=)Uk`iQn(X_LI#+dUHC&Ukn;^YO3yDn7RKL0k=c#jf_JXmG>I0@a*)aPay$euydaynWV}(!G zdBJxViT-K^d)g+>e>L)U3CA|@_Gz)lKH^YR*ia$^neA&%zhQLLT5bM(=UAW|l{!q( zkVd{b0G7W1oNQ%X@kAygSTVTR1wH`dh@BKnLQg~@vk zOaE}Q`go3Lzqzi;W^hpwKXlE~7=3a{_Oxo|-9v;?xaTwBo~9l>^M9W$^)zBe50g9& zrCj9d6O&@O-Tw2=?_NX_(>l&2#^++#W=vqWvPT=UMm1s&_ZB&lc4eiUKaww4SND&6*8hne!V8HqgCn$+*S*1$g&?5<{w_m(V6d()Td ziwS+gv#+$wIk~It+#K1n(v)fWUEy#t)83FLea|74q7#wac=fr-Don$E%7fWQzuw@p z+ME8G_7)CKKk8R@w5#a$Umr^|FW}hd6$&&1BTYckw65K7<)OVN`F&P{&^52RMvwzk zTJVA?`|IT_^CU_#Uj*EAi%F-x)LSy{52EGo-VNk=H#kWxwQ;$uqR%XVfB#(bh0%~F zJH1{@U}{c!(Iyw9vAMC#Lnl%tvJZsd+E?Ms%W*v7RQu22aAZnA5Cz?)s7<*C;*+Hj z#*Gbo1!g>?S~X1T`ItLA9@JLmU)4#2xPqwkXFBV;aIXhc)sg8FG@ur%ND{fih z=Rgl$dyszc4An3qugv-dP2UwwOXzk%Urn$gYxy16aXDTuYtaT~UhVDDwLb6sCGRb( zva8`b$kxCZ56NAS_Rg9@G#IxZ_ACoe9h=A+t$rg2tgZq}SnX$0v!Y{$OHqiBrhB=O zx7N~WWqE%cd@>4ukz5wUq7*Jy_W=!_%7-(lSTs?XY%VIm608f8DiSeWVWXdy1+jfu zvdCQ*sF3Xp{Wetw@yc|uBWeyVwOE{zxsWyG`m&^TJW=)R+5OcIaqBR$g10f{?PxDR zamhozW<{WWcV!vEzHlS)X$((K7w~^|Ajpd!fE>@G%{njjEK?tdd1`CcE7Fz=&(*6r zo>g;Qn)bzT-ICwJ6<)m7)rn}QmZ!;&I6@VKRtudxHETKVS<4=BNB6CF(4KD(**8mF zS0@70?AD_1dFpEEJ&)AbDVKS*T>&Pmo2LlfZ((GZaJywz0_mQE+=~P_+cFmX21d4$ z?*{Nw?_a-iBQS>&PsgTJXk#(bQltnUs*o=gX>uAs-}Ci6HS41x$dS>>1n0H2okt=W z66N*cYpOJ8w@jBZc{cR|hFq!Wc9UYKxSsm6BhLV%6qBW6dj1A_lPH%cuLS~L43;lS z%d0lm)o}nW6g8VwPfeg*fsaP73Q|B%XgQKkxz9RD>}h68t!dKtH27Eod2$$W@m73 z9yzfXVr#Lv^~EKCxBNl0-PcN-JV!1@vEG?(aXxZEHoL`)D!)2ha${36IJN%d?RJXN z^V~!^w%Rf!{Z%?$=PtLSUU=8Ra4{wBkg<_FNAIDkSthr0P+=-VCoY&9o-X;djV%_v zPC-1TnO#Nh6fGk`cD8ULoy%lXuB_g!-HKA)pDarcB=)VjvtORScy2^Q;$}Q|0!TEp zTb7(cvN@H3k>ZmByz08-Bt9!>J@cNt2o2@89RHW|aFrrbqwzC%mZS8({VV*sqCUtQ zfZd|y%@H7*vg&apW^DSc&ZbMpVGom{isS`lWG$`wKWA)0SnL;TvUzJ7lfrV1$uf1) zd=6W*_uAN$pEQp+usjjecje)cx%7RF!fuQtHLA6dY)vMfPi1cZ&@-p%g&&Lo!E0() zACowAZ2h0!kq6_lqB?Yiq%CeeIxBXyQ!j%^P+mNB^d}YAZ=44ll?_7@x<508-j+T~ zjY*AYk9IO>zlo7JK{k5KryztU`EJHQxJ3(nq}v%m@Paz0+>HA6L-CcK{72Uf=|AKj6|UXLNpgsFVA{ zoV8(X6^J!@yZ)(co3E_YH#zF-mv4E#>wvjKKNYKuG=k`t>l*?bt&RrZ=-1^Z>1}mV z3J45sckmE%w3@w>cRA<0Zo-U1S$d+&-AXA`jxRD6OH|gW$@%UsJr*@E7Z2EzN%$|e z++E`{{j?>w?kXSPS0{g2JBL-jBCFwKe0KYq3=s~oT5?3E=I6nIRdgo8x`xCZ?sh&c1o5D2$PONjmh0wCo{}Bq8XuXGv$jSKloa?| zb@<(>B2VGsV&KR+(csn9G&`z?L&rw3m2W3mov{0tG z>%sZyqT3oH#S@ECwjP2(I>|+-dG+)NCPVu811sx)c+JJ+Fb~rKp~ExH)ydHuaS49D zW@To_m~qx;ON1q8Vy9$~%Klc5Sru?%Wv{TQY<*Od z7z-@s#q6egU&!(Ps-HJcm$Gko2J%;%VYF1ccG1B}{BlTcMW!9iullrIof_BdKLU?O zoHsE&*4~^QtP!7y>laq|bh)8Rhg~o!5;16s8<-Blu%EQe6eGx*$~v8!r(wR-w3u7P z4?u$?4C>aJx-E%L4vBWJg2aN;Ye|5^KZU3Phm(|OYYW}JW}c{tExxrELWs^@Tva;l zD}S-PY@TF9LnSt5mf)WnvhD}2|3U>4T3ePFIZa~S5y1H>=rH4HY6@~9X5z29cst76 zY?8J1?#*kJpU8Z{Oy7CySZ}qwdk>ODaBPk z#y}UkNV7#U_=aD|G#*WVUwPd)wLl;}dBckYEh)v>S&Bae9zjhZ=qb9c=4{`-mj`zt z*18vAIhqwS3 zhynFY5UYP-Yq`WXlb!TIf#f~k5)$ZFny$%Te#}9N$>(s`r!zHms zx5DKNpy;T-*w)WbKV^*t&#JihNH9LDvD5xQCg%cUcf;_A{qg2x{iY|;q@RWZqt$pT~BPx>F9Kh>DelAR$qWD1)2A1r0=ryaW0_A$G3;uU-@n)A1_@M4w^ z{(_8_V==|ir)ng6s#dd6gr2`b(dFK|2Cd&gW0|=~n8~s~HP5CSthjj-E-v)j1YKg~ zec$$}O>BBkh;6knURFQ;;c%4BE1uQX8wrSQw#u&E$E>&0s5wfJ{D|1x7-JF(+!;#n zwVwJEi?S*9n0(he@4W|L&*|B(qF}o;5Kxl{9s}uJahwLQ6BZYa;lD)E45bJVkwose zgImT@aL9`Cz%`PF{YEwX>QCUT!X0Tu1`U5wxgYTG$O>;?JY9|K5)Eo%j-D%R(}XXx z(rw_vgg97dw9DWA7;N(h08$OZgi^e!53s+aF9XDEYRw;U6kj9OT~ZU8)2w_pk()ZhKwd zNb1chB!-3}>dE9VJo`Xfs)0ys5~;1K8xm3gvrm1B-wW zSUh$%`d1Tl%JMnPybswllxhiCvEt0fj)rfI84Mrd5Gh5|IRw4IoYwF%v6;PHx?(A zsSt+;h#B*LtuUhRq+;gev(l!R9a74(k=20{@e3WowGGPvKMflY?sf@(j@ z`V)cZHSPD^n3oL=mOVr312TBjktRrJE)zNYlbA*!t~&|-MSjFBy}m9^y%nCuO;53* zt@XsHz9vuEu*jI%zRZ6W9|fL-ijPh3QvVS3)}#$$-G8%x(f&dW+mU?mQc(P$RC7|Y z0Z@Kp>v#=%8sVLB--)Z0b?Tf!C~0Br?GlojH>6=xS0Og3W24EmGND{IdijWBPp#eg zW0R5{vLa7d2AAD&4_ni$Y72-XKx?KSR#^Bgp8d-iRNEF%v)@SO1d7VPt9vDd5jd{e;>GR%pHK|_y4e8GvsVLPwDiSNn4MpPnv}c}XLaqp{9&xll5B)r) zjjS&C&7Em5aQP$71hTK5j?N!m5W5kH3Gl69^t@}Hy{34w?jhehp&bU8b@B;sAhdd; zf+He&P}5{21CRpmM#DWfE&fqUYb zYm<_$s~vEMi#BY#_jjP2iW`H7&aGI#8au>NjJjbbS+YJ;R}bi@@*vtT-hnCNwupC#f2hb60)e>npg z8b)e0za6JbM?T`i%s?4{;7wB}hm)^~P;m3nYhdC)P!R)W?_MR(kS0CkNK@7%VFA90q(!%9Wu zRK(3SWrpPA;p$;T1^FRa7r4&NgV z?ID(Gh>(lIxTNhPjt}P}&Jsx^2S`MHCu}Y>mH&5~ZNfu{FTV!DBzfo_ zBec#Hy<>70J}VSf&jAzowqv;zn#waHfK&~^s6XPg060Lb);r>lIKD>$)7R8el=jQR zb@j#U-?miBzOiLX5!EF}MhnvTiL{SsI8Vt~He)XIvh{RzW*YS#F0;4I?9x8+H^I3Q zl++3HAwZ|ReF)+I z#{-;nLq5tD+rVRrLyN{51lR;H`3A`^f%aZF7;)~eQ6Pe-W)bzF^C!+WGRG0OwQNDp z`Dk4iN#5T=s--+>0MBN>{7UtQoU(|cA&XE2zK86db1$CbUT$8J?7#TZT2BFBa+ccP z)a+SLw|9noa3*7SY53BvnTLJ%@~9IS)d4CAdk4GoGU}Y)EQH4!HsR}8PQd`X91dA7 zo;A0LT;hU=C?%^_sc*Hng8{MTX`*NHgx(DQdB>0wk^h=u<62@}_!YmY((o-R*0mFC z7#v?pkFALB);LbpEgX`Cv1$!SlvsP8jCDx01Z#Lc;<%$%&&@k!_a1SA_)MNF2omEj ztDxPuIx5|b#(@afry*iP5ZQr)t%UMHwh-rL5`;3^a4cy!hMel|_rlaO64C|0bjdtp zg`^gIxkk1dVH73AY~#z%wdzj@N!*KQpbhmle)2&Ezt%QW{6e@xt7LiGa;8SDG%!3r z)9X4xVg)t^eL%?c95ArV1i})U7Hd26O8Eke>X(U)uL4crQ(JcJPFWG1KzhMX%riS? zcPyCdLqVHg9CUP5)URXDBVwUddQNZb!4%7!gc&80(Oa8VFM^W&5-E}a!nGX!FBCuz zMWa{Hv$u(6vkmI3r0GL4o>U>!^OgIMLtT9kvC)YSag1yh+TLcPu`(_ckB{4eEI_y>vkoYRs2swFv3as(d0nE9C|5KRuJmI=GwB30R zkU&lKx6?b)A0ph$ZM-LzP#?O?^!TfxLKm3a{AF(+-Ea%9LNCt0BqYq4@p?eI2Gb_b zincNz+R<-WcRnxSXY^z%%X*=uyP8I;F#P7e1@ZMu9{n*j&JvXe{~zB*3v;k@$CItG zZfbjsE0S2Z4xI?n^GF6a=h}T>MMVilgN`uOecZs7cJnNs?<<*Szddv%;YwAkXR5h6 z-477-7wzD1P3_@5DaU%NoAMy|d5`0D3OLrwR$#)TIMCq#dISy*4!E1G+@+Z8w+i4$ z4M)Tw{u!p#O&gO^MC{{Hn~_>kdU_O5&WGIw(I%3R!~az^1}Kui0>)_kHN zI_f$x6_0S%V{PMBK6PkjnwK0(Vd}0dn?*TyRv0fd^c#Zr1%zLs1QSds^_=l{E$Do% zspH>nqIiWGJ>b0>qjZHDox90yCe@a;eMYJ~U|KyXxkO*u)nLQBT88x7L_ev#ST>&lQSbL;|8Yk77<;QxE13}I8HsSyg_1-cy|0* zVo-ke=@3JHN}|EosEOer*eF-Cp_-Q)SQNkTFEz)6Ff$rCT96Y_zimZ)Oa)oN<^Pn?3z)I+QN5Ch3x~NHuF84RJFblj6eB5Vq%7DghXRnD*&A?`)N^h-EMY_ z96l&$f?0*Yz_Im^O2{2ss&|u9Ot!x}mR%GMOleI$5(Yj6c*H^GgLlM7b`e9LT0!x^nGx*(@bbDUz z{uhu+5SY)#5yIbfSE|{Ine)*XA$El?bh(&0RumsqW>PNBF4zng3$sH}9EN3A3RBq^Ql;w9T;3*c-W&`gc(rs@43(pTT z7G4_AjV;#QMEQDMKSaD0^kX6?#Y@&}I5UIJPpA$H+S)W}j9jQM+k8z~?I5LD+2O{P zRel1sNbHCSoEbJpr3gFe%)~11AU`8v_ju(SB0N!23X)!#Y`b{6P?+}8&e2-T0?G1M zwk9D^q29nG%A|_VbdhZU~3zf=*G~b`6K?Vl3m4q^QjH{A@8+g(z z6FS2Wnb!}olw44L%UQb7&ef(iD~;Pd)`wLM5NawzQ^6+RTG!pks0HT9;jGnI6bd}Q za}*@0on|@#`hZVIPkXQQqN^&3 zY*4%WNk#EGA&oNE^WP~{e|E;cSKE8IrBO+}0Xj(Ore!u{$#ks$!X`jkO=RSg&3?A4 zR|f|iRHN%BuMSIJh;lc8{4zv%o43d^XEh-_Tb+~l&^ZU@d2$h8Ag_&pCVtBs9Fh^k z)MB1y+!p5?>1@SwDlCHm>IDL&vstNpOQwoX{ieWyxC4JMjHP+nW-%w*kKAr0(WzF< zMIku_x!CSUsog;{VjQ7Vdx+ZeacCIleQNpo?3M{n-QzuYo?b1zHaO9{W)&So!16#Qbya?>Z5nLE%X2 zgbbQQkmtSj5B*s@<{ByWML@!4gC7;h`W3A%uKLh)8z@Irh^mqr1L~G7cLW4I1g$ML(jI;$%23L1nZ<%#lt? z0jp2LQ%`ZOM;z`+m~7xAp&#$lKZd^q%(stG!bxnm)a6{u0yEV%tF#s~)e{xm|2X@- zs5+Wzyb{OX=ni2%@{(QWmKKA z?X#PqrN&6ylE_jv9Nf1!K%Ppw)=)>3d<8TS6QnFXW7(-VuG4L=bM=vCiLF}7tk(;Qjr>f0@_zZ&NN+IqkX^z@7zX|qnm`0i`>WCWFV zF;ZHJVS6Xlm?V3cNFD8{6s_Al+0A+C%SKZI!jSCO3L{D~tf}Fi3tkKrcQ!g6`V><7 zW5Ptp*JG&8P1S@a4kl46l1R*c-v`XN`WLcgSm+aKKGTfMkrh7I}Bo?HuF%W zWU}GKxefDNkN2(WmRtBNG1n(GeOfj#RCO)1*ab#33!ffV8t3s2@kLW9Lu}<8( zi3~eH5g15ns7^U%9HMqE;E=QQt25l*k9AYda+DDbYRBgCrLZLL2jv{)v-y0&lWqGI zu0mp4c4w6^NLOC9XkHOzOP6+f75leFX9IaVf_Bt2ql-KAxzp29*x_8qUH$v}|gM5fqm-5)2 zIsf&Kk`;${m^xs|)vn)5+uQ_O$8VpCkG`tKbEJPHpo$L)8eRSdFB^RIN;gzpjnX;X zsxbUteVaU^VXHGb+f2u+SAqPmSues#LAm}=uTm=AC3XciZ>YsbqXi%jKY);c2G`sP+Oed z&VM!8+p|u8d=*4xy#LzMWw(Y|8M>Q(yeU7cUe+GroD*`p>n$BQ%cQl7bsx-6SH-q> zLe9mgRjW*BX_vn!_+u!Po)QF%$czLUe4pK2ct>C5wUpxPUg%*MuU1i44>VJIm%;ZU zwcr=|HhGjA$a}H<&!}nc5x*;Ep&E3)Wxr_IDAPay3~N;LKm1ETbFM*|3ch$DYf#)m z$$k>bczBgTO8g4mU7zCi$(#2CoHT(}QaGQU0tuK1XXjLkzNkB^Oy3sf=XmFvC-sn; zs6o3Nj`x08y2hSd_r107);Jot)K_oL{aU%8+EwwH>OChgMO5>hA@3a|#z?#{iY+V^ z^GkRh$v>0Cq>6pTT^Mo+U@qmbPUZ9cU?beXxW#c- zq+z#p&J8d_Ca4=Cura1H;aRVvOq%-s`-L?xX>r#vBM$|j{UYed>h=T)MTXAm8gdm)@^If{S9A!RN6tB5R}4lJ{2wwv>wWntxy3G{AUjIjr7%+m+{b8SkjfpYT06 z2J%iDI`{_u3)L3^E9~#vaZ)I8dKQ5XvB721iy84pycZskS8?^mA92=_ig%#t zHoXF7O#i&PMitIy(zWMB@f#3(k2v0`1;#ta(z`)K5$fQjBh|hSr)GpLl>bn2(^dde zWDh>5$G%*1VO;;o4Sq}QRe~q=qE8TEjHz>uZ!M(%LLwUelVa9>dIueU_%dixD}NzD z{tF@?XSn8rAa&OAD~WbtzjA~)YA{WjZ z;LD_bY4+a= z$RFrW-V3WDTgz@M+KBCDlTCk%;92Df8EyiK7`P-uqE@X-MT0%L{K=jO2qOYHw*R^pSfw)T&zP^aO+ zjVQvoQ6ZeK-aFqq(#zlOMReu}ePY~Jk4rVlQzGo3Peuhpk32V*J02hukqX5|UvJy= zS1K#Y2UzFFLiPUbe!IF#|IM395$i2c68Uvh0z@$RBc9#}trXheqm^eY#d0{H0%ywv zbd#@q{2y|B4r5)3h)w{XW;V;#FS0A`<12|a+k#65CP<%e0mR>sFol`fo;$Ilk&{>pCsZF z2fo&<&VCPg$<(@s_AS;E2oK=__b0199{qX2$5}>uk~zWP2APkHdwXXy;=LNh6Xlt?V#Sgh26+~wZ8g(KNph~m$Jz{PhF4W6X2&kFk~ z+7cb{EU^reMtgNF^@`c{BE>^F%e=|M#|i!J8=HeUnv?d;r`}3?K!0+A#u<~)`v_%% z^vDKtV`k30#P^gQ67JZ})Ur02#iuiL94bA9xiSW<8g)419P;AjB$%8$)3{}o1cBPv z)|~D2#`P^@@Xd_w;4_Qtsg4Ov#V2?`g%AIYDZ7-&kTNCOBt;|2DKn}Sw-X+y@`_&d zx11%HW|Z(H`{xm5urhy=xw+!k`1n9JH{w$+KZo$vYjN)L-=#loGDdusr_5!SsyV%c zUg2WfE0Q%K>4hAgLx*KAes)IHi`USbQ@0HkH#p2et2=8IM0mGiX{+2|;F{^)>=IyhfJ}DQpGlKOfIKt^EyO`^c zH6O>cf|g!_a&qO!VRzMbY@!@ury;Yu(qS4veCbfu`MPnh#08gY;#X3|Xy)4LQJ*Itv)vy_KG zWa`RA!%K@v|Ld>V%bvelzMIGQGiKaRlzM%{8Z6Gzi8PV z_7TSy_*TPD^6R&`&v_xMJ1(WQ*bNunbt(Gu2G-I*re|-W3r{KDsjfs1llPsa-pGk? zA{pbxVyU^yR4ec(AugBW#*s% zq=AL<)f-)YU)Wy5Y}f>`?iz4)hW8&*aHZ`#I$a0{bNJsdqKTACN- zYdrU>)});q9}1Io6zng+*RtRwerRMMA@%*yHb5z<*PG%WBgO7@AqnG+a>X}Ti2pb) z<4VAT{&`p8jM>?&Jynj&;JtMbcM7$Q@SpEKW5($&sF(ahG%*%C2&+sd?4V<@A7UXITy;+18aK< zYAwgy^dCOFI>-68G*I!bV0sF}u2i>_)sayXNxFnd#ZH9lM#OBDi$1S6^eVmEk>*D$ zA^IrBMD@T!)a9iWY>6L8uKdsb^ks%9ZwmihewU=HdcHl1%v+lOcRa{J@I`oL7)86$#h{CP}3Ftb8(mGl#6WY1*IY z{D&K((|&3A7drY^I#>w&v$tn#MLX?oq0CNz!lZCyKQVIl?H2>1hRq6jR-4;$ugvIZ zPP4PVmI+LsONpI9Ioo)VnsD5~s^;#y5hzU7+k!eh)2yJyDh_R_A)oTDG&2ciew_2S zT%?Y6H6s1>)JGw~*p*^60ZA7eiVrDIRB?DBea}7sD*o$5|MZnM7_F*rS6dRBo3I5< zkBJ!M*)cK+y?JhAjcjQRHg^n`XwJ!t+Pj(W$07`lo?*Nnad?Oz79D1ag<2p{FFSVn z?t7pbA#d%ej7;>+Dvt_7Yq~oT=SA}(R8bWgfzXGYDMR3X{D-Sw)@qo8H|?myVoHt= z4g4Ir0}p#FY=sEHe73({{!vTDzyxQ2gb7aCXxnn99iuZTMfoweGv@9DDH?)Z)G= zh8ad$|0<5#EAEZ>-J<;3JE57U;ULN=_w>`<=ka^D4`@SFbeN|9{B?JnM0{x8@&aub~PlVl`d<)leZmX=fVgff3_OaQbt zXm#fr5Dv&OzD5DqcuvF$rYhw5Z0&Ap;BU7DB(C!kc*=>yq)va!{cf}6__hJl2nS?o zU+3df`DG71;v{=}%&&RqQI{CK;>z-T#8IF!Kz_IQtFa$lKlqL>Xza@$XWU1e+qo%= z*V~b=&8$tO{p4LUha+9A9&xA${{r|9AdKb(X8aTUYV)KH>oOxM8am$Gw`@Dzjmqgg zHT5^f@gt$K7vXP?UX)~96Ktuc0?=A55;u~(s@xaZuj-CEg&p(_a_{xzEzEq84_}LT zX#eGHNpD##r0+OseY-xWKmabCbOC*gHHyrhK@j9v(2Ex(SvRg4?OV^?Jj8!L;&8=W zQ=&-cEeM6;Qmv=XH(O;}%}j#>&T1qpAS0Px*F*E#pIMef{>YY;W{MktBRy%()x?vYp_o_WI;)f58iUqQg4=@pvSRc#TV^!+ZZuxx!E; zjeHtyYv&i0ViL-3N7d-R+xarnK*%quBJu1jV<&4ojMXeZsDOcNcz35kwmK6tZhYCk z2>9&Y;@-Mco|SJ#5FZK<8GqF-u72X3AQ(f`qp$nntFZ8m7163?c)_y{fWg(PJ^U9X zq$w;@xk4|BKx9<^>5|0~vI-NC;WQBRbClu$I7&&iN56gP#H*~YbrlnyICB*KLmF?s zj~f(E^eI@anQNbyeicP%$zK}r-?8X~5w~k=?IxPdC?7^Mob7cTnbHpJL?!+xU)6s9 zfOMhO{bL+swMb4@yznwca39gE`WnB*9pVvEFYsrBqt1;lDE^kSfWxJGNy6-ej{iUt7M6kGUmgijC)EOFq#q9D$LgLfdPiQ^xNl8k| z)iLfi63g4`rWU}uVe$oL7wvR7qN3PmbFUQI!ER~lQ?eQEsMQ2Py%OxSLJ!kOjGRi% z9P`nQt^J$Elf;M!Y+9gKpVgv8t?hy|W8168}ZW8dDv zLA}^)yHF&&to0+2ECzPoWwdiJKf$P?d3)LwNCT=Coj0|R+m()u`~!wekzFm5uvYf) zMs|oqjx^r-)96dN`SRI3DE<9OdcTE-JH%(1GS#(Y4P4rf7v|NwS+(1faUNNWzCyuR zX_jid%$4qk%8~T|``&a|FwjG*HoKX%l>#aN<1!hXa8tk2r46)$wZ_ zx&y>OO3k8lekjYainuBND@=pwb+y6fW=H4~;1 z6o@{bzsCJS2{!X4&Yif(xv8=h6H#k^Wq$h}w4aSvR#z947bZE|SJve`mPc^>;~Wn+ zC{huZd6r+}m~8m2yFrrL-%Sy3p5`aTZ*E}Pu&~K%$yXmt5@C>*e;H+I+QYsW5EOXY z0d8-Nanf|6=&01PdS*DfLP#=&;u5HYn+k3<1JnC{ynkiFV{}Rn3!W(X%vp;bXrZs~ z`+<6>H(2o};M;)$w%oH{J@70)yFgP5PYm`BWGq2xe5o5UAHIw+?7qvqngCF)fj5%% zeZgd0iU1hi4;6@e6^l7QnLIc&Cu?9XC}-M2(nC$jZ?AeD9aWP8ogbt};c{4z**se|tP$B_%R!X%fwy zPprC5GkE&Q+3tmO{Xnt%#oc*!T3-6LuvV?BVTJR3gL58|DXurV&voV&Xu9&yMQqo%&A z^3m3+4sXwGGi`?MuZ!WS&FN^bD2+d zie;+L-%*$&YtiQmQT0$X7PaRS`3r26=|WodtckRAZQ=LM+bPr{M`zy%sxypECu~og zy|>as$vGWV2Wc~@Da&HH?cvObdjLPbhh+JnfaO2EGSphw2eW>dm$~K%nub^h6@Of1 zI|R2zy6hNfj%#g|sKW+2IJl4_Q(W}=UQ9#jpE?4LbH02>irCc<{AsN38`NSVezwOn z#$P;{aEw{{WUcFq`{w^wDk`wIu8r;Hc&`GbKZXeg-AgYb4!2hIeTbECXVXCkn#dcy z``u_EN!Tec;Yr;Arp0sk4^Q}s(o}Adc|lj}iA*C`hqr^*b9rMP^MY0HGhHpu3@M;7pe37ZAgsdE5zq6|tM}_g9?FrY@XoSXahn zyU>^4PLrj6hL~PUISH31R<7@4-gfTK!|;9~tUGWOiM2K}89&23%{!dQ%XAj~wCwE=cXDBCDQKLgJ&soj zTkV_k)F z?+%>!bQ=C1fAq=_Y$>-K$r?E$%E})b5#j_NGD4*Axmfv}g5|hhp|WR8m-KN5XfeQB z6mRz8R;rVJ%kn-A!um3>og_8U>x^@AedI6Wqzd(lM+uT5ZYyAQMu;akU1v_VG8HXirtTAoEH|L zsCSmjX`6)mFJa(w4eQJCpI5)A{tr`U`4t5NZT&GoDW!%6X_%oq6p$EV=m|QcVJK+? zlz50?2&th#YJeHKLqa5^m2RY46a+*h-gV#m<$l^{o%0v$-`Z=RrDFw_F2y|;2>e;4 z5O_|Af@QI-Q42)e;n=S*nb0O2rksX=WmPBj1sO(#S4fXefcd$6i$RF2*SeHU z&H7i6~)m#{K>`TAg!Co$UiqaH{*mN$tLuEiiWT4PTzxwm{SFEDVdesyjpM z;BToae^L6>W_rs7)Ox=&O@pav9&4jYKRC%{S#NSSMBgm(?{9UeVrDM3{CLDbf%uzoN7`bEl#`JX!(dA|3fSW`7>EGNA2Z@#hP%d;}`jaZs2v#jpTj;%m z@~@z4drAgtxDofRrWox|bJ^}FA-@Hh4we081aVHc)>KOkOXaGb(QSE5YgFy0J0wVD zJeh|?{*qMtk!yyt~YX<>^w=4>v~j>;aiyvV;1^LmtjQY=B@*y z`-)q+0~lz_-2hKe0wGT#Yz;VZtXYmJ4X!GfLA#-EGY@PtA|34L*+)1BbX002SPVaw zv})#-{z5a-H%m6WzsHjy(+6mKp+$QK07%e(N?`{8h-tk!{xF2+peX*rja}-Qd`VU7>M) z45{!xaoO|1j$-mjT4PDm31vC4aXNZ4tv;@|`XmWcl^|Z|)Wst;jS06W<}t~`IxbQ} z>%v|nB#5pjWMX3bwq&D|a4hY%`nikN5G4Pj#h>Zd>+tfz4m&ccwCDBp)Tg?)iyZDl zSe^X2R}Rn&PVwgrP-=hg+ZRQF&UDNMmLu1Ny()q(d`=>pG^!#$631ay7b^e#dnG7d zB4iGz%~qYMeW>hnP8QLp;_pGEN{*pPYm4h;Fd?+%m>2=}r4He)!5MqeStA+3 z7M;0WgH2Tmk1nPrqKkzsvfdW^6QR>2GzA_^dFR_cZdwc@vn;=tVgab5ddV=Mr4mui zH+8El`BIZRCydcT=b^wv8-5Qljo8*H>9G3+DM-rKUJ-EtNsF___TW_iP!Q_(O9znG84}r zdp+%o-wKK}|4BSSiQ#V%pW2N3$NkEJ#vkD*UJgei@v?>_++6|^e3bk;006)=>7L4K z9)>j{^cR|Nch(J^`*9n3#O`jiqq$nOi4kaD|1D-_2P3^26f&6_nqJa1-_hMBycXG6 zpYeE;0Ta?tdc*z}PLW})6+-^V7uootIZnk(Ry8-NvZg6kH6`QuCc~>f`5TH)&w1}d z5)Sbd{pG@nb+n82 z)3$9LX9TOMoK?|zH&wq>uT=4^Jfp1)FLT;eVRUDMt_;V<3D)@nJ2*p55dZ{2rI|%% ze z#T&kg-P$4c^dD#$ zik5DRl`&;5p|EiDIv#fB&v|uFmNW9US!QqTCJDEYk;)eG^~_jZ8ssBN*D?#IPpd%T z4?8>XJoZYJ?Z{9aAsNQVk(OB@WmJxffbe(2$aPd!iEQkF47+%JRQX77n59ZzeARH| z>Q{DC<-YizrkYVC>SC)6XlqhB0S=lda#@4vNBaVq(amj&k8G!7UaYPKw&Ze&b9tZZ z|03>qW^sKZn3%E8LZ~ z3N%xv@X~U{kOU9Ld(!oCgox1f@-SGzHByNB47ZB8-JzQT^V2>b$NQHfZ>mZbDBNwS zti;HSXBD5gWK|>lHJ#7?W^$$wh!70g~7aQEJbCic(P9cUWaN3xG%O^UGL?P2*W=K zG3K_Hb9#z)Q?ei`Aep@l7i;nvTTj3%!IRA(=**0WpBS~P*j2^*SluB`@VB$EEx$Ss z7M$Fp4~U{1{vfo9*2rgiAl~G|B}Jxs17>Hl@KILzxp2Z4N0O5B(JJv0)Pe@@T8Z`& zR~cDW$tmV5E9((sy3*rgAgd;C>c~;!$H(f?fj8zIDJStVs5G#LV`*l(kMv)snXLhob2g;_+a0+KOm7J= z_|rG|>kdX8BgbVo`ho~~kKGGR#;^2f6zyx zABJ>#KFyEG`p}^!d8ng>R;hJm^?>uNjQ)rTbeS8>chsFj2Qpd5kCpZkoOE2yk}cuz zNbIGK;W>nzFO19|MPxRjQ~r+~zE5-!#+2b3{DrThErFv2|wXAoIyLmcWTeUXX@;D_7VQrEVJhl%`N0JHQW4u30mu; z_BO0kMx`=9p1}V`H$QOoBIj=q|pQXj66pSkf9~8Foa7ZS~!+w;@ zS^4-gPNs>04itDc2Sfe5+c1QW4?dD=f#9aL__}g)kK!L96b%mu17^oLJeNT%sb2kF z+g0>w{(aYH|IGZ`P*1p6EIqm3jB?uBzh8{X zwJ__bSh0m!u*D?kw2!ke9Gq%ehN+1QI!v}JXS0huP9>pEGLoZ-AfyJ}ix4l>eJQBc zSj{Ky{m+JS^h~6YvL9Odrp^fbbrwtk>#B$XN=>FkGP^~2XV<{3W@bj+wMADaxjn^- z_PKMJq>x%G_A3>|7(^R4Z>*e2aVO}0nv~Unqdjw;v-GT**+<`|(jw=-pa=os3I|!Q ziEB=tlTx;E!$yavB4`oT-O-Q1O>Yi^n`1Gz_B&#lcyzL-6{kfSMxji@{xXq}??apL z$$_#@ew-=y22pU&ORmvq{I&TT_!?GQ12-_Lz0km@#WgNKtx_e9y_W6fIepB76Q z2lJOKEkFlB9k|8i1bo7zQFUyCl{s>|OLw-lb-IywSHRIeD0;9oLB%M++pds|JG9B_ zS+b4Z|+zZpCZDr0zp9`yik`l)(`goY0k_u1%$mGw%NpO zcBzPSoIlpl8Z-tu+uqRsfNr((ZqlcCoh>VRR;oboJ3Ab|Y{5XTl0;)~{xB~sECr(l|O8BJguQdI0 z)Kv-8dtX+|I|G-t8ISKiCxaV7N43Tj&D4#QM|M&9R~3S85aeGC>V(3n816xkCJ)c8 zY;$yoM(=r&RTnMqp4VGj3;3-VuP>A1y_bT855r&T4lo*|7! z>xjMAfghCjiv+yu_D02kuXfv&D|f%`jM<599mHnk+n4-NG}TiDnINEMwCiNojE}1y zdg=c)w%`esnpaACvRBttg8g(u9^wc5In1xe^%@8BS|q^->}H;H2QtB?O8r~mv? z4aM%0?ENW6qJy?S`A^MBS`K1DH^d)4yswbkrchb@NP1KHlWY7jx;gvmKaUbO`*a`b znvZ>hNr|S_`Xmol*=Iwl1|kPo=Bm}rjGLdjwD?4bIMEDTbJDN$5khIdyUF|>HY~3N z=8idNb;zNXN4;C?NEcJ=mYPg>#2$yf&WSaabM$0{{d#9$wCcFrB|z%LzW}PH^YCr8 z?Tdp4?P(9`i`mN2zW%xOkWvqjx4=?vna!b#1tZSG|x1a4R_o6g&m<44o``wrk(-l*-<_z!OJ zr2_0kCOzo7WpB}{Nz<=c)W9{;!{ultWSM4!|%l ziTm%D!f@MADJ0%HotEX{FOi9h1S?fugzcf+04Npn-#OG;w3mr;-_(h|wE3SKg%12} zlEZ4C2y$GzzUqV)7S)TJOOk4x9Z$3yY$)Y!QxDM90Q-a#7l!bcTm%Jxp2yiFh9-+c zO^Ham`z2m4{z?V9!u{kfiAa(k}4uUD_# zn}9s;FXi|jX}U8C8jNCdvOT3!Wa=4=uc*?yRccNS+ug4ak6BcobX1*ZmVB@wZL_mT z%qZDXs+g`Mw+l%XhR0B8Gsj8Ewfy!LZc$nyvm*L1)Ye@wl>ZL%%Zql%OH*Iao6g30 z;tj1N_K~3kC{P_H=Do7S@1=|Osr@(_c(h#P8m%g<8VXVmn9gq?R{AtlZZ1&sLs+}X z*Io1qVkL&w#^Zw1$^a2q>j&yiH0jCE&O5+|wPjqLIaA#v_D74cLI-x}4&bT{W5NBl{ zyl+FsI8T=5j>*>#qY#F0hT-zUcsr(F_2?K!W)9V%Hrku`?l}JJ94#O^D=tgxPsFKg z(ic0C4XQr+Uez(3tWZi!X7Q3i0iup=l1rHL2^YYN(u?6aV<@GUj;9Nsoyf)${?4K` zFS44bjsL$T>3Zg!Jd`T3%D8M7A{*;#Wv`=3zx42}W(-xG(aaF};;5j1OVd!%=SeCJ ztL@(EtwoSD9|#`LO8;rsR<=)$c=l&<8)X(u`ZAx;v`zFRf_d?woL!|d{Nq*~kd0CN_7F0xoPfFoMWo}xEAui_O z^+GQ?`f%r!9DigRrN0N6om*@tx$Nnn<@^LPvI}-mjm5q_p+F);I2>^u+9SbCO=Y~o z>dg-D%+-h5?+Nr+!E0Ip^Ld`S)(-u;v*Bm62A^*Dtu&mK`2t=^XMXzLB{rcGTJ=>_ zb;?2nX7pA|+t!c%-VZG`=@UXU>1`}L>flwwFha;qCoAJeg=K-(#t@-wa(e38pg{8cjL3{G^fF%1=*OUvC-kz z`Ysd+>%i=81QFj;F39ma^W&h|HrGw~592dST0Li*fjqsn1P+_TR_P(pnDcDA3%x{6 zjLNtkpLG6+>3IUvv8QK3sFGfX_O}kPqO9}%=^KH+r=StXH2H3yrA{`_1b6<7J3tgE z{|~MIe^OYQZ;-L}UXseq8Ay8_QyQ*W zZiS0>xZ`i=wF<)>ps0)Jrs`o!m9784+;N3K1=;O@^+7l&P6dQ$(1yHhnQ2%|>ylIr zlcAwWD|GRDApQB|S#Ou>CUa&1Jq_0CG;w{Xw?jh2e)(hJuVe8FnTCZCgUU_kgEv*X zBFHIYbcBV2?pj&|a4$xL^$Sn!kW|y(JfZgL&dB<9CJ7&hyo$&b0+HnBsCAYOL9RgP zuW^1&tl2Zky^i`#iIp26?+tP6#D$!m*cxs%B-iAj!mr$I>f=P$g zFNLrRbX?QCe1Zr*da#c(JmhWIe#VPmU?kT#F+QQ2A*ZsS?z zf8!HM$ox*Q@o~zJ`gH{%Ay_gG%w40zk@~>@-V>ym(XxB0`#Q8k+Npu^qu1ak#ecY!JEqQcX!f%9IpScx; zT^3u~ui>?SW$^`g7~<8`Z%G`W0v6&-I~bSl;?u-28t()Yhr1x(KM2!E3v5Fo_N5D$uQjuHM*7SfuU9CRG(ZGe;^L5DPlgQL5KRvO5 z90i}>x9qD=wG7|!F(>>!Bga&%^IZb0rV+32qY8)+k83d?b`$;rAjAyiozN8S~p=ZGEK-G0hikOdn z|Le#`hO7N)o5!dz%gNjk;=x5#ZI(aD>`L(z_c5Q_!9ma{^^L3@<3y%qhX>Up$0(v1 z{i5VIO>66dT@DFcPU#o3Xo3^=M`zt+gsYlx)lK0-gwIAm`qjd%`fF0! zvPoZCZh3<-HFrd>9T?A$nLqpG*^{d>Iqaqj#SXB~!+|L=x6_>QU;wdP%wiTXOhdm& z$HysLsLy?qtL7SO3dH zkHz%NNQtM~Lh60xS{eH0CyINb2R$ysnf^|$6lYTZhZg`ibPNgtH#zl9iZd2XzKGau zVW!9AfD7LD$g)aNbEc@b6KT3&6`=arw9i%Smk9i@8s*6Cxq*a-riYhH7~tjVHwHX< z5vP|&Ar-=ysz##{`ss>3yV$$I@_Z)&;v5 zOpU^SA(1}+ok$|YTxkznnd;`XnM-v^nD)NrJx5?V{B14eciA^bN4;q0QOuMsd=m03 zHSMM(AZ8K#lhJ7L%o3z|#{{s$Yb7~|2h9Jv;QaLuHOV5xS2t;%YaCuB5BskY`5ZTD z1y*C9%=TuFF4HpWnF!1i&=&~!Gh1x0-F4q``4+tk?JY)(3mmvP9@nN#uI2iaT55pY zh!L#`WV$8X-TwQZ>PY2uoQ(CvxN(l(+pU!-`N`1s8}5MG38DgJDNfiu(z1eaW-4xb_?5*CCMlQCjwZzj|8WVDBSXcFV!^&{u@CalT5c7M~6W znfS^>IkoZDgUn#!SA&b7ksp#?b%LeC;GkxgyinJ&c=4oz+}i0QmpZcsM9fD_jXYk% z%u2aZ(r#?o*>i~FeQIj5Ud4WnV0>Sl zBr77-`+GRn(m}`1SQ=HGJg)d5TKS+h>DN?d1{-IgmAPy+a?P;egET?DLsg^drQT`r z5z@JA&A{*3m}?BBU-`p#ea(6O1z?fv<;R;`=)>BlzHSjSvgrY^>)AEB_d26S-j>>K zRYWf7At%sbR7{=&xotAvVGUEe7GFOo-i|gf*XOBslLb1+?J1p`#%n9YByAJJ@q~#4 zp=2&00#7>@f5ZA>!Y4STRD`8e=CSjd#0})4wYwzgnE-`t&v@4DyTF`b^Vh(#rZj`( z8FH@7krvGLaF$RB8qV^Tj@H)|71Dh*`%pw(0$H(UQzLF7l_rF3>k-)s1b)mCJ@6l7waS7>B-Q(~l9XJ9`8W#i#&aD-hO%7*g0 z6|u^is68e#b}a2)-`~D~pnHo3#b|D@1$?WN4!1Awn|FVc5Z%$a9|rjQ3Xq+N>kEhV zd8gXzgX%BD?T@yBghj;Oc+}yRJ1L(JHq_`#dU3jdnQi_$ldT_2(lel^yR`gE?6UXf zM$#LZN$c>Gi$kBL6rQxSGqZ~f`3$0?EWJfBu1o&cGnBIeX%p(>w%Rv8AxcUTqOK|} z%$ki6$+<`SDRJuS266VVg0cJF&~V~jhkxVo%|Idktew*PLO6;oHNBTm>W9zgUamr# z3vP+5x327x6Yl`co|-`$A8sFeo_2C5u$_~OJ+)|ivKX`z*LIHs01(IBw+Ita19~Tq z%pdJtKs?;Fisx=t)eY6nAf8S&+IW{l%x(*T+)I7~h1~TkXx=e;Qf?mF%~!bPsgr~r{wj;Sf!tLPM~rlxC31OHH$Y;cGo!>*_gi!NSX^Ao$OF=l zfN0vb5%|ccUFz@KM2680EO|ki%f7=FGkDsS8AYec{Ff+A`pC5ta!*ad zb&^gWnnJG3-2Dv@HXJQOwaLIQ%-bban3rT=P5ShxP^4~zlzfng_!Yk{GWe5$=?TXt zhO85oHUl(uix1+SZKg%_4R@f5kO1Pu#b}I#ZyfHyr#IQU72Pt1h~4YdRbCwX^eD-q$q0j~q>fPRHFEJ5!8j zstCg+R{i@VL}__14|BT&?(62Hm(y9!p z=N({M^+B3*(QDs2*U4QIHmp+N6Xw=xSrqdXH~uhD~^c-&ITyo0rt0u<+A*c z$MEIm+1=YCw_o4?zTx0?5&TK*+2F4ym9p@4^j+&f&rGhStBYF=>7d}h9c(Dlpf;<CRulP;9r_~mAy3)9I9GP_+yM^c z$w;&qDpre+usSD4w(plxZZwo_SfSEI?ZS;^rEGkR%&*QWTJ*?ayxHIpMpfn>5F0sg zT59uCNLd-{0`n>7$kBLolzP<))l0*(3O`uCE9ua5QHn>G^2)QvK-baV@YeV@e_sFU zH@ki3XoHnLoc&QqUr;$dTCl#)(zWELw+ZYC1^Z(8hnOru6OqBK4K>I&Rq4%GZ@%cs zQGVJG!P?86MZ}kf(J`el>_hfv%iU{*|D=#g{Wf@7QV)-mxU=~1M8VoR(Sg35;Ieko znS~--@p97!&}CJTY2jXpLG9Oug3mi}rd4sfwpye)LP;{`c* z7E7MXg)*3!q*SrycE<~{$l7=QkMhMl{S=leSp|-K-2q_BkQ9eLiNL$_7IjxiVZQawO>m2DNuksLnN4;JMs+@+waYBpA-`_@(2m}(! z4xfEmUZ?Q&iIDmXH|a%jHqT``ls$!57!;hW>x;XbqJSWODk{kcJ!0#pLt>iC{>o8Y z>}%U4Cz)w&>rD>}R{ds}5WEA}3#vbZ_Xa7%K!<;E@X+tPDQ;Jse}2F><{oMmu~6WWcGXg zhW;w%+vTpmvZ~@|pGvyHisSyB-><)z5yggUiEO04&8hnD_7waq<+_(!Q188#arZ$< zfAh$zJ3z-l@h+sl+$4`J^%}ot?$RuA2RMOM{@o6(ywZ}HY9TUFSDL|AdyKkp6@h?N;1xv{ZTm1d66j##uTx&y2N zHGbFM0Zhz4yHNCOBi<&vlXVZj-_^J;=u*2(jns;B_PYbHwUxi~wLW}Ad=GQ>4l+SS zd**7sD41HZ&{TNYLXD6JRyXl zk1j76R>Qtcbn{+5t_FgO^X@Z^DM_qZvh@_6eR!r7z0(IzKQ%)ZmmT}ZAvjKa1TxNJ z9O{2WE`kv^0yj#;qB%&5#xxzWE{ z#FMs#@b3_=Tu5hW#$?vq^q?Qp+T4p;M+)$u{#)_vM48gC9-ICX&{6B?);XAY$@f(K z9|~Jce^Owcab~Rc9oV@^Q1?89PiQ-ueraFmr1k_x><5vhsmOx^QsQ&<|B`5zz((_1yuI2twrsbK3c)G|#`MWcga_{YuNqmN+zdFD%_x+Q=*^ z*H}4;6hW8YeuR@>^-ZRv%D!)!r>m)l|Ju94Kc#bT@5+H;+WUG^HfRp9IHSuWW!S$2 zw~UmUbJfbx0Bq&*)olva??`!$)V5OpOC+l74uIj0N?3~&=96xFjq#qkVBZ_K11R7O zE{EU!EZFPmhdM^NK9}x3Ve{aB{>=DT!19nIkN#XoYI-^(aOe&|rQ#}Cg#4Gd=u!jp z_vX0+{3>dKNUcabTX=XBPEY*LXl4%{_;s0MzknxpFFSe1+K=B4^4~X1aa(;uu`wR> zWu^81n;rjsCe=$|y8&zHQW_YL0C2oMrY%$8Zzx>;8&t>K& z#r?>Q;o4N;^$V)Xv11q_`&J0adUc3K7JjoZ^w`qAAZ@e~a0b2;E>4gsXhmxM0RB=S zdqr1W(*}Rc+@Im-sca~hIAIMa-L0_GzU{02pN@-nwc`Z4MUP*wv{|5=%WJ_4i&Zc~ zR~S;mQ}VHNkFFvPWp}qcO#gBB|w`!0Dsm zNh-OyupzvZYsZt373mdbi{*XIC4u|2uCOKu!0P`jmYt#UAIZxqr+JnAnKjBa)w0Z(BC)SCYqb`) zpeNjOzW=n}D$|S6FSf-P)+Fij?b8E|c0i93_n#{@6WldNXE|B%HXpNHpKGVQ$Cle! zslnw*tGB&ZQhS;q}Ci za;V&acGSm_E-4sx~LfJQv~dsATI^C2N(wtVLd9)%Vu&yxjX`x~M?=|a3U2SM9%r40#2uW21h?>79C27>uHEruL%O^YYNEkI8v8Pg;@Z=h zmi8btY$_=7}^wfLR zHCB%)u&S))`BX8AoYP13mXr&#avvjv>(e}S%+z3%HF+OM9_@Yj0T&pOx&`d2x$CAy z;0N~u9L*Al&R;`zsZkg`lUP@WjL%9rEN(*}m{8H0Y4NVO){(QLZEyIHxg z5&B4X>NqASuVbt4a*RhIkV6Na^k>+U+@lv=xur`%X5wL1NRWvRB|Zs!!14#HEWXk5 zmxt9^7E=w;k8qe4!oumSi5r}483Xi$ou177LB1=5c63l%%xqTJMU$Cum4V57R#gwliIzD`f9)OM$-#{47NH`n^pn2NC-gNvkkP>(+!> zhnoz0D4n3ONQPvYJZ&3FX-f=`^+R;n=DLnT zHlT`j;8{J2CjP6zzUK^3Az5 zO5vTDtiEEf5aj|8sJ@u--y7-E3uf=VwC*UmGfmj#E*pMkD~7c;BP-@pg$a_9rws4+ z5xMjr&-gb!<&;3c{{*WODeaF-M9%^HAga-!zK);kJ>FC~FK+y2Y6R+Z#rQ~DTyFVi zw11%Wt({qiKy&v?&(WwDg`0FCAsFpSR<`- z{rib=L{01)eoFcBCAT$WCV`(&$i!($jsyj@uB~0r|GmnoO_;_Vh&@Am~Hl$d+aEI4Ds}#alNve;PSZT z4x`z7dH73LO~RB*Um!kh8L~{-{u%Fb`{hKB zrA{q?OTxbIcStur<5df|9qReGaBfklRZ~C5C|snywFFB<0tlO$M#4GXdfjVSmKGtp zn4~Xt3<}T8M!J=rIKDE!zg%HQBS1YRaH`X-HkXv>^;Vod1qlt;^OA&rg?u>~HH@9M znRq^T2PjLhcKMB@wfb|kvm`hk7Y@=vWI?i~q=W-ed3f~~Qq>y9Pm@egiHZHjBJ$Y^ zE77t^z5!(Bmbt=q*(Yg|f9BZ#C*TLyjhIG>M{)kkBOq?Kss;c4?3*DU+0Tp^rfwZ= zcvEykn@llA*SOa*uc6!}0%A&(3srty@S_c~YdO?C6uq4Aso48je=^DwVwiyV+RQzb za)gtp8f)I0i*h)>*TsRKJG)Pyp-;8CNo$m?J#=&pmaRW1=sv>UmXDWC>W|bM@+Ua> zr$=p8I5km)I5&MRwe+_(b@tQwD*7#yj6G&RMowKy;*j$884NwPoc?EVV4gH}mg8En zB$IE4T)dO~XVivjZJA8Qzx!MCDjf48m-SJ54!&alI|xTZ?T7}Q`XMX^t@a)0c=Fr? z+n|u4dU{a(1zD=FMG??tw7<^WOk>N!B0ipw2I36<#mL{W$e2Ewe1jb_@6R?FkpkU}`>rFC zk1jWfUp2*Bl=puShq#E%dzozArmdg_$UJxqY!CzPx?+1sjG+_Qg0EK`HN7jzjH=AfB3 z3@`<1d01>b?s(0XxAj?s?ju%nyTfCs{tnAbwOhZ>^w*dk?*zT+$IkPreVBHoC$+^%O)-@HA{K(+ zw@Y6Ctxkq$#V+TpoZJB(aVsT-@}6t_Tp@hSAeOJFEd0qZAj8n>(#COSLfr{tQpFyF zsx3Q+_JJc~KAc48M2iDJ4)!G(QY-MYrfp|g0iB>mI8g>6xmB5$bdLYCY<;<1PayzL z_l-!aoU{Nh+gBgCMg^c3sN}srAcMxrTx>n39YIO0qZoJQ@Y!yCmNOdxJTn_5Vm^eB z-O|>dPaYXXccpBZ|1mXpG`DJ$V(PC7$_wk7mF+;jnSuDHP6$%j^T=;bO?nj*r72u- z!bB%_xp3s~4t{S;t5-9WYm7Fj2}L47vmEL4^Db4{2tk3Mg&}^%?#x!Z~^bMlbG_ z%8;PKbcVSic+0GAk-hl-w%9Z8pXZ-R!HcL*8W4F9%1Hdvcn38 zvywftL6T`s&P1Jc=jdNzPW367VcEshbR$l1e$0);r$WtPtF7Ivil3qE`)&MhEyp90 z2IC1rgnLpflI-@$G8EJpESBH535qOQ68JO$tOREtr{Oo8Te#O)N@**Jq|&tvaE3%F z>zuZj+51>QA{FxpccU~xB4=R5o`BvO=Y|L$S&XOz_v0dfIG*+;Pz8v8DS9lXF2yk! zV}C3IVl%mM`?MUOP48a5GjN`Cy)-rN@{|!Z)v;;QN&av4DN*(3`Nm@vBDtYb)i^e^ zzfBJcY##*NOly6j+brf{TI z7I6olXvt+UO6Rh<b_x}C#H(|et)%)ePb)Z z4@Q4%wy4u3i0L+w)$w|7Z*%+$9A`b33tbG}h1}NWm-n+q#k5`wxk#tS-2X_6E;z%G(5Uti{Mi#PcPVp*mob7S{FF*M~#;;LxAf0woSUSjnt}-uXv9i zf?yM}+v*uc&px!ff7>%*V_%rwhUXlR4p){rakE>&r){5dt*{aeL!lzk@Rc3ntw!A5 zThC%k*7Skb7o$b#-|IWwwC6R-QZC!QWxFEoALCRfLe*^c^WJTJcXZX$UM|bb=ttXI zD|4`FhN`dcuPTnsHhl-Zb634k=^~Uqo22fSpIve7{HG{_QH$H7)j%5Fa0muwFID|_ zd*T$g(JrTM!6SN`g4H`saCK~c@VZL9?C-*BO_DYJvyz(FSTo%KXCx2%t+-=beiF(% z8D6ks=r<`fm9|Wi+57b@#q6(N(c{dAaskDQWm5{HO~0?26AH{jfvd_uuM@mTn3Aw> z#Qok(v;2HBPE}hmilY}xG?VgB?{|B;Gw1F=bNM1?NfX@SNR;pPu*r^inb2Vr|1orA zHM$|*C~IZOS%-zM%glpQtNYpgQ__SM&2RWG&IW!V3U9RsW@`DwGKaV!u+Cl){!%8- zs9vyFhV;K)N0RDuywUoBAY=In+_}WOx9L%G$r&NL&`Ddtx41tLn!6*qViS=c#auf) zojH7}?k8X`@t~wHKR`+W+A?cCSog;+csLGgQ7q0G0-9w=|j^HXnTTgz!ecX3RPOy zQo@6}%nYW%En`7(kawCLdN#=%B$XvEhF$UzLR0f^l z^jYmh+pq53|M!w!0%DFD$ZnM9&CuXgrF(I_gFP8ezYn?n#gA^XNI<)&$&@xt$znztLA}%0$Cmjuy`-X+AiTtma;w$WnginraIOG8RVL;H?KAakl@!a) zPwP~ap^~{ttr!#EQj_MEqunarSBvcl&PwGt=)X}*;?$6dguRy&-(Y%Elk#i(xrFyK z&dk+sGq8E5gzww5csn#2a4!Mrx{*a^e>p|QE>BLN-5H1*1{w&Y_xdsaU1CE5lw-F|yCd;igfX66Nkr+pWue zlRY$@WZ!Xuy++8}y`!2uX=7uUar#p)jpTPQ6hGoCHXntw^`nkTkX#gU>96OwY?-i} zrSjWC7D=09li0Jq^Vg@?0dcR_JDQn|l*8?uLP z9+_xN3#PXtupIOv^2x3COcJk!Awto=yK@L}k#JM4Rh{h)>+~0un*R8Y6z2-Wf@ahg zbK<7^&Av$ajgn*zte7zOOkJQW;Vo^7NVHF4WPC$uQBiiP;RKVzG>iEl>TSEYJWTIr z`#aANfs(h0kFO*jYu%^jWHWO4gExN8Pz6CwN=73*>2!Nz5>t$q6Nes>t1>LtW8o#z zCR4@^@#_A8$nT#wCt9euyHlPFPXt_JF9RrD^G{`uP4_W*sf;k>fsW zYrSHT;&je=YMZPmNZlb9rF(Q&Jw< zs<^C_sKA`dFC)$V$v?S^c;y}Wh8gCgXv9S~d4S;PKP4@%W*=+N{OGRrOQE2#@N2po}O4Y4p>ix9Jwwj*j)Uiwjt+>ZDTa0=Tg z+m$d{RNo=9Xn5h2g7~||MQXlMXS4FHZCHk(Xe&<7sVI40C+z?xP-RUdr95R&cH}U0 zv_;>vS5^>Km3dYHVHfNqb58)e#}0m+lFTzS)hwgZR#8mze|MiEMZApZ;5YQebtNGP z>fR^Fnb6Qjdc1OY;RXKF`NaEUs!W6jwQiqd*%N7@=g%fm+(VV*@LXdPI{U-~hzW%1 znHJaUZOQr%?n9OO@09?I_jK0QwPl;~;Ci}W^$_*YVg2Ec?|$^ufZK-|A-bpWGK|DA zD3&xiZnbnZp^Zar;TrK5RFZ~XZ2X@wcKt9;ew98vJ0~R(Itw5cdpqw*Fh_~3Io#@( z$T95R0ir$ES6j<;|4n}1zEbH*{P>Ll6XY-$Ff}L1dJujOaR2qgFaQ?ts*t|pOs5PE zf`N#^hpZ%xJ>^bgUNMX|TGQzA(0B|;5SoaHQD6R1vBN`UfBw>I_dfWJ z8WpTJJGf#i&Se$Ygj)<_ug)ASBFJQOrTNqU0BD{893~#|w0ixG(lf25&@X`bQj9nzD6AD`c08Ac>P`MscG1VE*e+ko zLNiy>Kx4u%ToP09RFOT=^~sjL{t}9k`1(>i^$-pa>`7G36?ZB>${llk(`5E)r~X?= zzJQJqXHar-RUoCk#Hpa}!|dHp5-q}hg5kmO9Y!bVapRKLEZK8nNbY>3&HY*QiXMxh zJQ|gDZ2B*w8)MrorC4ITEzc+y;_6pH!Xad_2Ch-IHf@Vp{gzw2;h4o(RIF@b8c!ul zQmSN|ZJ=w$8p^71aasAiF~8l9SAT;0P+AV-Vh!=I@5|^UV5*L{IKQNi*y@crdzH4){{Shxp?xNxhCN~;_bEDdNgC$vZEMsi zymw9#^w1fnA=aFn(DYiw9G53?wF^97#hxjBGL&Fh+8+fe4PwIyN;5a0s8vU5F!OP_ z`qz@Z{{TnwYW!IWub?8U;1d&RN^1+YTetqk6;X@IEJK5Spqi<*4$dla z_|XnNpo)^&=G3BbX4Z<#ziiOe0$wz8f{CfJlZ~Nkbq6ri-)uv=`X|P z=R8{#d>(Y-x~b-P^Z4VauCbq=Ggn$fmguWuk}RcHP1Ns7Qy&4caf+1$JXDBLKjie7 zbhL&Vo{nz#n4v6A*qq(Lvg)Y*EHjU&GDCgyjbHZ{BD}kvExlj<7}R9=_~W7Ji7V|% zI9XM9Dhs)$+YiE{OqY!P1idvzcbTp)E8+J%*4*oAEq?5dgK6>g1ae-Q;dxr)65Yx7 zB`t&1KgTLc6|aw1#+%Sw!y2iVX(w_iR$B&Jv3*-svvmIe^8roh6e<@GUgT5A_ALt6 z?yG9E^7yaaFX$j)o@6Zie&kcn>}6-k8sp1*r3>*au&xT=1 z*LR*f;N1M&-OA1@CH^v}K)#DH95KP1UQRZq=jtxXEuycKFJTFLMUMVcM>BKis*oZO*G|KA(8&oz*gL}CLfhq9crR&%f8j20GHxM0 z)y@{^K-OPo?o3id+x0YW>#=ezy#x#fX^T%LOAB+*whUJVzRuU|-Ou}P1N5mB3wupd znjU#KKH3YT;Kj&Na=+Coh}AsYa-1_IF`9EiCfw0&$rkx2Cu{!zR$T?nvXOw#w-{O- z*qyL2`?AZbr2J@SA5;yPSi2lpWGO1D8~A3Jm1FVkRZG(X#^NkhEq6{{5iWta+4yRl z(40+7TQb*okhm2#8s<4Uj3R|hxM|`p1xF19!;Fc@lm!s=12}BrrAwVLOv=_31bs6C z*z3qiDnuHlUQxs<(BtA*g2*+?;C4N{)DZibEIu~F=(h#!?BWJAi?Z;rkD{u-FM?50 zDyj0!4pH4D#Jb#HRTSb6Y+8v^>`)P`dO=TB0M3Qv;Xq6o86i0XBE(cD7?O_wW|0%b z^x}!Bl1GM}=c#l8rKvFw#E83*icB>>QhiCn;?$pOe^Kn27CDMS5dQ!qE;8+=H@yqP z8hWsQbc`TnUVKwkRSU`fp<{c?<9Y_vO~A9-nO_zqz_*O1U)2(gF3jwL*xg;bWp!7K zXtm4flsNPz3BC|4pkxa!(wjj1eC zbF;{(h}>GbK_jZFX(~&_aK4X-d?F1OJznacpPh_L^qZs##pH@|L#r>eh8Iru&uTLG zU8`|$b=fmSKTu(qjGUaJF{#{GT`+#ZV

U=o^xuOi{9tt~;=?8CbAtc?<4CVI#N1 z1yG(dd`NHs9$m#l&{&Q_Z$$$oL^4}k1!{$B1r+gFsw(i&D8X4$hd^mcXtYmz2<8>6 zwR1JyDvEiUIms_^hO)yoohxZdVjU~DiF7z-yPIhd1z-gL!k~e@pae!;#tzDeb`Iea zXH?y78}Ekxa!QnQBl^gGcmDuWS;NH`?}^8&66(qj{6W14`-TG&>?xn&41}N|(wH&LE3aNrYVrNoP`(afqOB1|+T=X%bCwVzlIXF5Y$9faQMDINc z{oyzcOi^-0av7xJCwcALm*o64;XP@{T9}Jl-g_fZIp#`XnogIdGQJ0~YBF)#Y?xWs z5X~`^??c%-frlK5lsd{5CMyHPu10bw-^OnEz>(2PBI0m4)Yw)fR==lhDjsev72YF? zM(&erQ!jbCx+c*e(w=G}A3gs7P$0_V7(&HPXl%XK_EjstCbhLa4)Nl130!%A+{@i? zmW}gV;9el6Mr9h7uW{u+(%zOrhYfJ|uOpruK+U!Hcz4No$QYxe$3yNNr6Q@ClqPC& zUcQNn9Aq7KK~p6>OfbC+depy(=3RW}XL|Q4? zg)t0U7M1f zLRN~gm?)XT7>^uZessC4Q{E{D4srS9RO50t7CYGW1b-icvapTb}2j?^ASH19D) z{{X;lu%Sw?!41Z8D^hE}&<#?yJkU1X9JHjAO^?#A3)_mJa!XCfInps%H#Zx>;*g_Z z5IyG1rRgTna3*HeWmUO-Q?9SNuBr5nsfGp>JWCKq6*Nr8LMo|sw?LYixP(I`Q`lID zROO+K*ko2p-*kOhT|3BJtf(dlQ%o;(rOS@!<8l*CIL_`ml}7|iKy0b5gqa@`FBd4r zq#ort3|r+2g91e}I=0sfx-xiT5@d6ub2dP*!2kkGya9wop$yE>wEC#j`=e$YuBv!&@omaW+_S2$)f(uv zQ)O~Yjruka*B~CxrNKrZV$2t7X$R{;h zM&o-uwE1-^NVkg9QjRp8DA58LCHx~KG-AGn^azTFm6C2wv+E?3Ek~TWoSIAA9JwJ% zR6HW2IZ>mk9SfDoLvg4$aotmCg$kb7(&lZnr8l%SoRlqwc?wj`z(F7ZhDj@(JW+2C zN8}3{W@Gcmxq^?&8!ZK&fq3H71&z~;cLq4I0`f3gnNl~jYBv7>;Zl=JZ6c18qFo78 z({5{-e4dW&lN9dzvqebh4~-_5s&S_Y_e4}IR?xa4YsD$b30Nvv^A{`A+08|qoBpiS z4hfxdW1gSzzZ9<=XJSkyWU**$aiHdK)~3Qy#+bKD32j=paaB_7uG9o2^-|(IC)OQF zNy`$25oPYoCX`zXl>wwK?yHLg7l%lXaBCct{p!UBiR% zOQ4ZCMitCPL_)}HK7>TF?Q~l>+xN3Xh?VaUgUZBmDx0v`LhT;QI5CcG6v?Kkj1b{o zxe*ogLm8!8t!lNRb+V*3^=b7^l|F|~c1;t%@e9Eq_jpW;V7BEv%;a^<=?Y$wY>`uW zD0|8uf`IPBUL&Y+u2(7TSQE9?l?y!9>25!9407;$lcWvk9VCXzH4xQ#np9@HOw1h0 znIe&cj_6$z7r7fuSb?pn`NvGz#KgnJRx+w8{N=cXAxkNbX^iJ}yLbaiEwij=hh6vl#t z;F**vSW2t$tDCxds%Eg3|c-i+-ddJQDa|GLzvuS7;QER9oQ(@e4rs9u@Ml6 zwSTMyh^BBA&GiWc(7{v%1xWiGfsG6;AQTpWL^A_AWB{fo`(Zs&-9P*czsdB6BOLz# z@XhQy*Da7Pj)P)yM<)i2L^fyP)y76s-@0;-RR!qfIWa2KA}$~R0000X&9gA6YTE3~ znuN2ew$f))r|B4lEU2(>70I8gpK-B4M)Yhk@K0nT7RUmI5=Wao##B;hUuE(*`DAoziVKTJ@g|G zKXUQS&X^d==BKd1dNl_MZxTPG^G%rN8x2G~3v%Rmsr~}QEJ7enKnkb=00IgB0Duxo z0ABzA03ZMWW&i;ImiP1lC#gtYKmGz9`X)hMt#x1Q-h^!sAY3VklIu4zuqP&mv_#&ijR==Mk zor7R%_2bXx?s@0+IgrQ8moH^SJ2`TTn%X8uvxe1op${Yi!x~VKD1e&yu7#C7FOWqE zxkw7_BnY%nk7qCA8PLauo&*AehMoj_Ie!>}cuC+pIe!=q5_k*W&@r^08h8um<%Ohh z)4*RpEI|x{ih32(gt#tMC{R9G6s~VqRSY!Q)RL%tQ>98G%Sq~xo9lBkY7$D{EdpIQ zDlVj}q+T0}gp-S@laym#xiNXs?_nZ;>~n}`*?C%-xE{XUhSJ%tY0aTo07{*#t^p%-J$O(@|Q5NtzPr7 zhjcas1gz5B?Y@Szac-8(>h>+E)hF?$Tzz3>4?^53S|1|XrL5YDyemVg5B}6}B890X zEYjBUv#a0z-&0Dj@vfCOq+uxxb!Mk=Z8O(u@SLh|7~;f+W5H^79vG|h#~JX<>~DcK zBAofdhJwcU5?(l(7FwPJdpUm?Q3Vi_vzPIO1rT;}{xB3k1b_r@e;@)N2xQ5VDZ2*C z*ch%zMb2v~((Ox9()E=<#+j$to!j{(NPsfYoG7K8Ul3>fyB>sfxv|#{HqscIl2;5D^Nf)M(M*yg>Jq69x$XIj zn%DKTfAzX~y8eTrg$G!jOsGwkfU#w=b{gC9N^V2CY2iH^!d>X0F2v{M8N)=#>aEPy zA4k~qO7_)LR&f$7+%x+@_0>>8vPV7BE8R##UNEOMK2g$MkxycAx}3XnugvGWPh~?| zO>-w%Ra>E{?29~H<|=LCWLGL_@lq}13!+eBFEu^B0JGS96h^@A@l4dkvQTyyOqyGu zdb(lP>lj>!b(0XpVnnJz-1AbH-2BYN#kH!P_r%i#FQ({5W+^?0Ie8TYZMlsu9L=RW z@maG&(p2G@VoYKS+)Q`7nNr|J+Ebqy(*!*fjv|U?uf&&OD(nb|zj8fSlhMR66}ajHsilVlL+wVpCd$n5Fz3x%!?dxlU$em-H7X zQAbe3UH-dMJ()C z+R1;)3b{zv<6@}D{{ZULlxc45pcGQHz;pGK_WuAWYhx$5RY)P=r4r3AK>(tcw*pS} zJk;GIZc6BVlI76&gOLwFp+zii1ux#EwZ*mD8>`vPs#Tvnxcb0|Y_|fH@hDcb>qu7i zEumCqFNH3@))p5x36^PTbv-uz@!!loU*lIffY@#yDuvq5>GkUR>IRv}bN>3tH)?CV zJBC!HFsPST|!%s*@o8Hb0m!LLkebF+=%p>jY*w|emZ5C-%`L$gtsRZ1mtyM1t z-4k0n{v2luC;es_n*NeT7B$u8bQq|e<#IONO}$3s&70EZ@QWLP{{Y*nkl1WWv7Kz z^b%A2}94m|UDF!42%JTiCKR&;ak2eySk-`)z4$B7ZW#c7DWzwR)UwmoP{-prxd zzPYRA8oB)vrUaOXsah5$H7eYm%*q#5Uug{V-B1-7OUmt9_cpGKmOK+Q#Mr#h$KEh_ z$!r-GYM^dPiCq<}%q~TVJ1p#+P^`hSyi1}Lty2dilKPG*%^k^GXke^DrEPsj6sk2@ zE+CE*O{t zQ}pfyDHN&EWtqw3g2akzI}(P)qByz6(={Z`?$cA!Z@ebWRbS8;TpfxrshZTu8Q8V1 zNff7RX7wH3QjyA_U89Is(<4pz6NXAFg5h4y3R}b z%*qH8{nJwn)>-c4THF^BiOwPDY3nXS?h6@Xc$&gx;^p%Ukjtn!#mUiG*qOR1=vzi$ z+_xB6bp(eX-KodB)|kTNkwW(LMSIlrM%ZhXSBVz1L}7WT_(o`Y2?tR`HXG3-@hRR~ z<*Gg5S`_h1$w}g<7i5=6F?^?aYIHgi;k~u)I&KznZrFD*I6Pd$fhs6==PB1gbCq>f zYo(-a8SzpaS%`cm{(gV-{L42Yy%y;b#9Wb7QeZ!%Z{8lRYUY_mK!HO zCXZ_KL)f;pYRPC8)2gqYSNg#cT@<&lWt#^jA7)EeOa9-^EAg(5ucj(kYHF?oo+RIA zYg$ORH{hxKX6PH!D6L_sTHs0IQnK-Ekl~9R?iRI&&K2rn?4!nl?v+8hNdkV zpIV)&<4XC8yJSBsrs5cE%f#W8DjM33+=FBkl=5@g8%Rkg+}GNnAmZ{wINOOljwx}! zQx&oImALVkn{5Wf&QI=#(Mc{*$4VD7Nm^@DI9cO`9J)Q_6+QEgq#(H=SfBU})^jwJ zs!VRfFj7~K8%DEQN&AwZ)w)MjRZYbq1?2{#vlwSH($f~HGKq^JRjN$MGZ#s$VLiy7 zFNSai-YP^Zd=l_(fHC%M{@PE+0AzzCB~L>8V>Xy6OSqX*D`vGr^}8A3)uhp!_~mhQ zzx@X6Ge##fc{p)=#uhlir;a!n(5{Yqmk^{k1mG16VsM$Qr$NdUY&+}~WU+ZdfBK6N9M6d}vDtyad+wtqQL%&m1c z?2eH0=~kS-Xyh|3-PTs~Ck6F2 zEQZJ`mZu}9vioBPDF{=o?F|POvzc#o{v(|qnYj$Ly8i$H%Z#}jkkc~ONa~AO#rS*oyyrR1@)|yVv_h{M_43cE8 z2$e$`a}yNBArf6L7T~%z9t<>e%!j0S35;U7eR!9x|eHd;*#PEpoy`9qU}O$~vt(L&nkQWY~3mL(R`K1H73 zv9GQ~T|-*h6pm+U)^gaoMsT~*-S7_Fb|sl)B}F;0BV76+um>(1ZHlfp)6_YR(JeyI zs5@AaY{Gd{Gj6F4lSMR;!Z}K7w?}3a4zj~d7K(4`9;1P^svkT%Vwo3YJ6G)urG$Pt z27{sXS5Vk$)b^rfpTNs0W|c#es_D8|u+NSBd677u8kYp(xXrmTA**Vc<=^ET^;Ke< zjCD}Q>We<3D@;A~Q_SBOoD~d9s_hVL!HlWiRH#-{G;KFVvoktg9SHQ#!@?$+yki_A zMpp_K=BO;;LZVI)#Y}_8br6+uLqi>?mvd+{3t4q8v66*laJ4j&TyJAfO4+MKh?8{EE{a4T^({(daGdIwM)e%STewdQEJC$u ziq;jqXESRSYY39YT{eoyOsZ0)$4R~s4sN_y*Hp)aX|k<4UT&AL+d{3T+^%Jt(~oeH zDmP4tp6=-}Cr&}(&MHNj0n-x=@_1+ec`Yr;9)}BKYnPNVQ<1vo2OG24Ut3}Zh zk5fMx1`3$IVMnBNN-4`(xOUd`GPupnR_QEnwQRy{pB~)kewOIRi*cgtW=R&6`+4jWZh$5RV;M3ARWOzwbozN|i1*Dc%17v(USoG9YV`meXc!zuUO~0HP_z7paQl zD(_Yjk^2!?$0gdjLV~=6pN$jMgSg(-<(QnmZ`x&BIsX95U{XW$ zs)@CHNaA8nR_YlSn3Csxj#GH#Ey3k6=}Vz1q8_KC@glYML{r8JvMra_3(Gg8soCc9P41OB zh8fC9E5g1~-r)`xM&tE(wz%wL*n4J~R5QbFsI(@xi%!K-boqHPwBu6xWia|nVO1)4 zM_^Lbs;d1;H=C)gyg@~+MVe%O=c5?eUdjV;a(}I^KYDge+Cjw|l z00000lmH*;yoW6sTWTFBvm#r@QBx45@QrwIG559{Ix3s^YAU?5Mr7c zG`Bj1*^8_rTsH*c(aOi7(M=feky}(QqqK`tNRI?aARu!B6=*`LCo_H6HsuP+*3O*V zt>{W?j{Q<;E<$q!8ZVi-D0xh4rEqY@tQl`gA-d~}ezKj;Fe_Q*gGDirm*Bd8?xiXD zMpy8r8p_pq*$TM2qwma=D8pychO=DrJAYbGf3iWc+^X7Q0YaWSS zsfOnD%BlYVqRnsnrXAWE3DGoDJe2(r8{C@5w(fY>cqUY(5|D1p*ueTckU#3C^N6O0 z7b8zuL~`8zM7M6`)5+`%(>_J9BBV+(0#JYy{{WIeAPAwBU??gCT7`o6T7a7rMF_lK zFDxi8Cl!`l09;6?_z9sGD)J%}7by>CkSj7rTgt~`v+Jsy2{SaN*I zI^%VwNQk9qI|8Q@qTmQyhH^svzU#v|_@)G6skMi6dL(7yO}gdqIobR7Qf?0JMaqsA zP}h`EQ(iHQZZWS6(Wr;2Y^*tSM3+h;9vXkjEkrLxSAVsdkrI{y+o(02HY{M2xOWsrx^8V)aIragtqC zjzhXE0)eNuG+A@Ib+m+T&A)GQIg!29GMj=@_)3jsa@6@&Gl)g$9QS6+ucC@^{i9d) z4W`yFs~VNgE0^nE@Dnc-rSM8GVt1cK!?5^r5~#>&$i_HE>}1K<8x2Y!x{P!taTJt6 zD7DN1AWaFV3JR0o3Krfkg$3l^7{-N_a*+xODuIOy5)?9EprkYdI+i6sRLu#*S{2B@ zqy?x~ZF`BhBDxn`_3j1sf`~10n-;?Csok(zENv);Q<0uS=Ta;44NLB(70fNDc}|q$ zJ(nt~y!(yoyN=ZvBa$-1DHD|l$TX5O5epjuu+pOk!x+#bDcmjv3sKa{gf933%TTX|)bAw=P*tgwNyao5 zB6kyo3Aq!vfqkH%nlc`uaG_!{9-?remkq?l_JV}k@Spev3vGr?-~|Wvke`YqzUV4M z1+aOW=W#_fz6*Dbcg2Hh;zav6Mf24a#6D;KrcOQ{8i>{BmgOewYMAvMyv?E6zMMKu zc&#Mw{fAY_|}G<9V3mC*=`^$-P?w#+;OT}ObnG+fz`QN{dO6e{+dFkNLP}Oy&uE>07J6|ty?%Q3;F_+U&8Hn0l&MM zA%e&$TJ(9p@wlff%>b!QpV?b7-_`PtlKm-_6)cmQl&P%~0-a7rj%8$TO>x~lR5LGu zEQ0$bJ5r5xm3nhIv2bbhDb}FuvEhxGRaGk=1$bl^^N8r92f|jTBVH`NB?=OMZ*!x! zAkc91y44h{B~{yYuFBm5{h+ECMBdDq0!HMN;pql&_6 zRQADmuHU$IgxUpF?q62*%f~TDaeN35T+gNSVYr&%j75Z4OG#L}QrvepTu69HGiuw@ zRZ+T<(G@d2Yvr*g+x@a)rDLoO_N@U^$>b**LU|;LZ<3QyVo5|LxCSEc`3a3t+3QnZ zi?KM1+^Go{?tKMHKv@^Bd9dD$Vh-mJ^fjen6uE_+P^?YLEDB5hmhibGYHc^75WiE8 zLfcKEw=KpK4CBkmkBe%B-l-vLWKj*K^iEMdWi9UQR??q+*hWQ?g!ENr_F+H91YeMSLdHHd6AirkpQ z12k568TE8-b=Z7?npxmY=-jKYzA&{Ag=<~Oq=IjuTETkqKqoR=udKpHYsS{jWBLDi^v!Us!{Oh{MjPd>j2-!EHpEYG=&UHuFP$k1a}S z5hQN2Rb2I-^>uWMvB4O?n5rIw{X4}OIU{SAk*`MD#XgX!4PnsoH?}ZUH5DF8VOptL zqw08@+Za$frse{*r){UTaC4gTn>4Ctx&ij;nR>DxcJ zs#%9V*B!oNkC!UkdR0t}ikGQ#b0-@7{mlO4rzAzCYKC~9_z2-CuHv3rahwZWXGhZf z5`1F~*esz+n(j(vF)9kl29g!DVYaI`M%j9x=f*zFt<%XHiqf}w8<7*zD3P%|I^0000000R(1 zD1dR@xl&N|lX{Wlr+15jYan)QKLo4Nl-<<)yeX69zX;-CPYdf#}^^HgnFmEs~_U!Ir!w}y#bQCgC`r3i>^%D4TPbuBlsmbv9Sr& z=4(ZR-fB(J&wEe0pL5+qd%3FiMmMiWqSaEGQ0jcXz^zn-DO4J&Sk7g=&)*!_=Ik=k z=;MH3&sEuIHOylCrAdsSL4?2P&x>zeC6TFN{&ioRvuRQp8i}&HL9(UI_9r5{pA| z5VK1wilIrxAjO7MF}aB%Flu5G-3`DSEi$2psIC!jjh2cQ^BCtf^8?bVVC;(>T~$NU z5o$?D;+WSDgxLb(xL)TZq#Lr*_j5Je;|HSX*;|AMm$~n3DT(wL= zy)hGpmZ7MSR{EO|#JD+*eHN!eB3i?dcSsWo)fVvBH#qdN3Xs{E&xopq*xZ*5D$1Fd zn00upWE~KVOUlCGoN@67qu#VOAS$mjYgW|B>S&VYPIEbk^+@V1*?34eC+eGmaicF2 ziISE@s&XzAO-6dQnQt(wV>5N8gQmv(6NHa-Qf-KshMmT6T&iRPSrVL#)R?5aCfl+F zuAr2dURgUap_cyuIHVHAe<8$&WPif^Y50X6hYG`0&P7v|KQU@i4C0xy<`gU&dU^#P zwcF{O#`jg#K9SWwEXN+&9J;}65c5xnu@tN_OBaNRhG<~bs47bvf{BHQS;(lM>Gy`+ zD8+-MZuY*2Rnv}AM{w|^DRvSY4P^~zvC0qKC8+nMxh|+^&+dtXH;qH(sy15WYMQ_9 zBBAm^f!#4{bd@&3k*6yT)}tI#rji6;+^Um}!@|B{?(lfcM$)H>#c(-Tkrm3t&*G&^ z?a(&gEM+tMP^mmpzg05qlzK>Lkr$cEZVe>F^*1qjO4h8K%lr3JO33_s#Z~mi&jk?K zNmzZzpF+@Q3Y*$c;uWZc$t=_)#PEbzfe{b|hG`MMh){1d$fVmsev0Dx4L*7uwn4u%)38d;g<-PJp%i*fPnfacvVY)mz zhU%46(^#Bx-r65lla0h`9yT*r)l^b%XDHutBB4rH1{V=kP!|hTNrc=S>$wpCi1aF$ z$heJmo5n5`LBd8Ul2E6*Vwj?*LMn+YlMGmK8&p%N?3;Em9!75p-QtN+K`ug|cc)Jg za(rhxHC~F-!s0y>Nq4Mi6}8;y`xSPdxb_~Yi8Y{Zb)7aFPl}%ClXQ#1%41BC&3$Q}{(47;&yW?4-jc(}^gxMJUCb z&ANDi7Loy20zeQ>f4%^aVRfrg5iZEsyP9mM_SbK;{U+(QEKF(pX)XO+gXFzE#}-*A zihAq9(O48>X?DL zhMZP4Q;}FYs^oH1)$|+8xk5oqRI+X1;UkxR`O>9H1W_&H80H%9j%&u7ha|;|5!Rv| zk7hfS=u0AmEv9aWl4e|p6sJNg#=MbD$v--;fHYqbRdPOWiFBeqGWC|ikyFJ@eQz7E zChY+wX*vxP!Ba&h6~bsL)NPQ{M7B-QMk)yun6Z@cw45%l7V=-KwlrTAT&ID=`av5Rj<|G}_*Fwu zTgK>GvfCv}XTQDhOpo@AVn^;teRB+$)6w8K1S~QgU7Ng3QP|7a?2V?KhPsZ(lt2JE z5(%)%wIkWeCc}C8z8;tnwgXp!#qHWx|UgKuxIk>k3E71sa+aM+HanKfTxfY|C zkIPwFl%iBUESGdu#FmQcm%vEFOsj-;Ns;jXdo7;IG! zT`emwE|hCaCSJ~uBu&V4k?<3fcPAE6q;VX)o4EsUh$f8)*Xi7pU}BF&{U8exi~2wj zRq-646<-h)GR%~|*i;LL(zrpLOjThL2&sakG1S~NjUiWuk_^3%y_6o(8s{cwQtmvk2GwKrEo!?&iBk0GjH+1LDLGV{TO}HiR;PpM zXGoS;{(@IHjxg=j7X0SfD?>DkRK=#E3}mCx=Q&H8!#K_Zd(`Q4Qvysj4_b(y-giIP z(W7Yfzy~~l0&oO?0001-0t@^QEkS$*Nw13NSdIz*0Dw@n6%?FGA#Dl8h@n^p1}HcJ z3kk$j&Ks}Dm=#K+s!O7|%6Em6L`LpY+~px^6%%rnqn}8Mq=K9){{Y2M{d0{(&2v5< ziOf5s_{vDLHZ@KbotKfQW*$#WIXdqo6+5NZHw{%zd&e-nypOQq@~JHiun{QKD12d= zyJ_fOXaxv?Y7pd{4A7hm$XzGn0YoOkAjr#vqX-D#i;DnOkOH-aq~KdsCh>q(NP(RQ zlm(lx$K zRgBe4k{ryDJrwApSW z9hqEQld9(1@Z943VMfbmZSOh^#p1IyQn?!4&+(=S8^t}*u-r`$-DgPRGa}^Pw2D8v zwp}5#n1nSqp>&R)?mY{WT815d43TemS|Y@ohNi;whDFIoy$i&D+lwZPKJH~sLXfFx zUcM@P#5H1?7@j{NhsrEOa#o%vThE;<3b3SM8mv;x$rR`q#o=cP=_;?uDD!BE8b84N z#~~gcOZrvBioS|3iOCPN)G^1XV!=d1^(2&e7%fJkrbR8mwAa8y=5${_CE}k_TodvJ z9byrmXvuh|)SC%zA|?*844%`H)Spw+FnE9BYn$W;ScX)%yqbT=hoNBQPH*6MhWx?9 ztV8iCb~PrS@*(ZLDdl)H+=0Ki+X4yn)Bn4!0@dQT;(7;y%WenalqGQfIzyarHy3%6O0JDGw3$FRo9(qG}(TTzyfG z8W~f%OC{n>_9nCwaAhMiq|$oH zzQXpW-VI0dTH-rG3Vx(3UXw|u>m&9NwSMp>e2GzCUuTI^x?Lx$VfGKTPV!{S;}Yne zx`w*+x=&F12->H4?>R>u zHF(2HYGof&%JjD#k;Nt4h~DKlw(CnEo3uE%TB?$Y$C8h}-O?A*NZGmcxcrk2WC)h7 zXT&-I(ug=r`pQ4Vj?U`jP=Fz=;h8X%LZN#B3ht%^ov>u6E zN%(8w^*l^*LUL0TjKywGHy`eP&XOB-g0arHGf|5({{VG(d`G9yZdFtgPIdK1A427s z9y&?GK0&m6LywLsQ1wk(p+podtqc1a>H7vHW^{2>CW#a<&4^H}Ebkw=lyqk=R=lBx z>kzGBo{h%rK4!Cue-?q`8ghM{w5dg;di z+Pvgc-XTs?AI&I-A)F&LpdVzAr%@(T-pbtAJ0i$xR~<7lfoV_x0000007xZy7&R8A zGhS3%+^1YRNZp4@mj%DZm}|KU*xISjH8WICN6SXvHPf(gk&WY|VW{Jx)= zrmHu}o?o6Ju9Jax+=io=Y{5gCcR8HyEak(_Gm3^`Q~bAv6)O$ep|k~Q%z~l1M|5Jg zE0s_;5me8nQDOr`?v|{pIJ%n1(W>IErw>>+Gby&m`&Cqqa+Jj}Po|Qh_-bm!L+E4Y zNVR06&M66sEP9&lOwSV^f;1woM#>iVUwCq(DZ2!;x;W*>T#e{_%UhAfjHf~wZ&1TU z?y8$ee9_%a8s&w%RT!w4rEsYiOTr=2DDL6zL`=!F)kC7>hQzHxG}yK`V|8t)RLn@i zDauhPDnq)EF>a((iQ#d~IWsd1N}?%ZH2O%2L~e>`mx4l&r5;)e&kWG-$<6N3~fW5Qw^5FQm4A6A3avNaOp~Vh(xjy#5Kj-o9VkXqN3#AQ^;BHEp-Yc_mjk|J9K z8q*KB*oIx4FeYxD-Vp}0Je0BgjU!oDvkOSIhfn0CR6(mK)>x<}T{Ul=QQ2gALp0H~ zQ+Y+iFoun96%g2{GiZ?#Nh0w`LC1Vso@_OxJECDIB|Zjcn>&q6Y$6n;Iq#nD80Q7d zaxY?GHpbA!sd0{16{^&hV@=ZR%O_h@9m+00m?uhCkWvn=8-_YFuA$AFn@AZO8q4hE zwVptkvaU@bH6U#2W=UUlc;Nhh6ud%@cP%Z$JiNrh#4{nY5T+hrVC1QkHMGenntbyz zn9C#ft4%ZbN92nku}b5-lFXX+*4nK~usz&1>c`WG8^CVg6#jbSZkd{V(C1&>9F&8o z>Kik%DQt=vjDyz`GD)Svz)35m9IRGjgxZ}pW4i%|7f+Z-YxAWQyn zE}VnP!Rdrp52C5)V|0D5ZzG*;3dZd^#|&pP&?~uy=_NKP68S^vcn?gO6{wC=#^QY$ zTB@OsHC>zZO63ET$&aSVPcmhhDdOVFx?NVMqKFfaHyIMg8p{%=o&i4|L2nd_lHhnj z#T7xCpYe%Fgi%x_!&{pcUWzJ9cJB~V5$VEIE`;hh^rK2g4d)a@S3W04n5~b=8^E|dE7=Fuc=-=11PUF9I0p#X@yV)qM%Kn1*Cxh z0;Nb)uQNTf3uLcso?9fPB`{3=?ztbPDd*gz$CFKbcmIAiPh2d&4KQ_ zk4_f7VwEpi6s1WAWid;NZv|8Ez9_<~t5%1CbfzcKh>G|=lRQ6v&+v2AwQggy8y#?? z$87s+36%ZHO|d(d=Q*PdxKlLddi)=x(zu-wR@ZnMa_3_&vy&j&Scvn}t?@lxM8hj8 zMI3-VQaH-oi3U}urw=p&EU2sEo`WgF~_EDAw z&E>6fO}FobUcJV?10;v#uAE{i9$`@w`nCZj3SJUKdCHbgIk{Fs&!b|k9?jJm9?dE(#7fec8GOQ=p;r{@yp_87(H!0OwN=i!Iq~KOR1u3=| zgxe+36m+I*JYHKTK&lk0?i;Hl@~FL@T2&WQlZsZ-IBNd@V8R=ft5h~2^v+ee zMrxw>p48Ui%X5;EZqj-Yv`cKJy1S)h5KyEiD(Mbmj+`l%Q!|^0bl(qzim=wJ{tkkW zb7UKgsqDfA9Mjd4#J$|a(l>--eh9t^Mir`_HXO{yOk&>mTEy-+*3_+9(5+^8({*$8 zRmr7*+@!ja1mu*W#p%*8J>}C8}MAZdQfXb*%NOp4mAOrx)s04r@03?tA069WH004jlfGlGv z%(%iq%2YB<;o;b>?bEeNRFO=#j;YKsRW+uj83n56EfDM)KF)0Wa$Po!quSm^()Fv< zuf)QX#Wreh$CPv8y;W#wmbw+ou{?z8nS+&~E*eXRTI<5XwWZ)p!zU0|1 zNlGn&0Unv2vk1avu-H5%C#B+GxjP-7XCTQ?F-{nhbT6-R2d-l{LqiL6SBjyZS1Z*)Qh_mLjHF?JBTBZ^O6a{8@O z2ebPCZc`o2yg6F$6_qfp<87)y;wbBr?&QKLwcNPOr7Nk6M}BTGl4{KY{*>Z)iV`Lu z{OLe?mvek_{{U!NZCxx+glZI!^HOXV7&;95mYiCAf|j3SZAxy`Zh4Kw7C2exz();PRdrV(jurs!Nj#@lV2R^6o$Lu7%yH&gohy;r190T+^ED}YI7A=yAIdt z^^My@Q|mFme5{v*YP)jvq%pqPQF* zvfM4r{cAwAxx+dw$MzRzDtuEX2pj#<C>LeuLA+@AZUkXO)j20e2T zZ|JE%*z&@UcMEhDU(eF({;7-uw5R=+=ZUPI%~vr=7rGjjH}$yb6hHhqge^yD&iBIK z_&vVB4OG)>ZMJN$`(;$UG>R*#mUf%_+P_21ZO-%DIO_h-q&_03WS8koK{=7ywC0LP zUCfri;JD1UQ!$wCbE#2LW7&THtV8Dn`8y(UOQc4MebayK?YLqlX zy1PMIKw3aSQi<=p?|Z(V@!WHsd;b#!qm>}b^mOvITS^{j-OvG(+1(?p_Xv_Zx%Tpn zHk6^F;Nf_)LU(9tJ11Lmn$FMEB6VBpeN9od(n^uu^Obgi1l!?(ACq{5f$c zwjwT{;D_m+;Q9ywxYx}215ICQ0LXcA|LpiNn0LtE%T6#+_bU;3&gFualwV8=23rLH zf!Bk;1EjuT!C=7veE?+RwfO;n*fZ&G1fzB88Ia$m@YDHS^{n&r>rE=Zd|SB0B*`Zr z7r|kv6B%$;^6YG7U!Kx{H~DO;vj$^+ZaY8V^bW>jjYWNaUeB z?go?N?q4zr*%X!Zn*Z8HP}INFk_&(B^i0$^LJk!W)LWm{LX4|r(YbX=8%XsQyDL^2jhB`gsCb6hJ1LIKXNZ*Y(blREhn zW$ZQ13ujee&~m>gcC%ve)`j`UZx9K3%GobuLgjfC!NCQwjT<~_ut`!F4OP+2ijiA{ zV6=8nnATVk3qjhz6&7Xe&`lv`bG}R9v|{qX3@@QowwWH;yP|-kVm>F^u_kk7vESXv z2?$Gt!q;u%6-w){ouX zq9YVXO@O{&K`FF*bf?B8u+=w$0N>iv5?nV zVR?ARy~2j!UwvXP`T*CTKk>E^I7?YNq#%0G%k`@(_udZ+J;XAQy#Mh$U_-e%L>Jy;tN`y~lgV9;$wtN}vPcj&X<~lWe zmCi1P7mh0By~U&WL9OI%Jn`p&`*2l6j|1T~OFpP|`c-|lDo;*;yW_6wTM3H;dU*AS z&+o{}Ys^Zy%IpAsH~rTtN?mJ8*_+P`6r0~wb{Oa?f!Xc9Tk{046<Kb9$W{v z{CEPK!^-bwbrY(#T)q-HzpGo+CO0i@&U*sHE;>x}#T4JtAj2izfflvdP0i~HBAgeK zruiPMc)h5&$ngTRj>}_e-Y!n|HywZZewy+4Wm5N%72Wfwg#%@w3PA$@I!#666L;_5{{#|V-Bv>n%tiN$CjTc$z6pP<+yWdo^{CMw|tgJ z;Q8?ruj*bE<1;5O>Tk29Z{^StcGJ>f9iz9t4oZI6{>j?Ez4BSc|p z4V@upjEY@!+FZ-G$Xt^EjjwdX#T(96O2)G$Y`-%=CHT5g+;T%pSzc%VV6W#zRQ9=i zHeg+ym}pF=a<80&x7KEt^Q=VWGQ0Kt?J%~VWc5pGa$XfYv(keWox?ITW`vbzezq`B zf{5Ju=5sgtnt(;1*c{8BI0SQqI3TSXrPcC>o$;y%U$YR{@8Pf?Jbt#hH)nwY9#ok( zXQR=>-=5Q>S7n zUn>qx_KYe8r@)V_E^5N>*WVj9J-h2@5QO%=$tski z*q^!FVZ`W5dc7~O;=&Sj7PXI8jvfX^@^I-w$H(t&d1pHd0ol0Q6WaN?~!stB&1 zuq0WjlksC{g0A{uXAt_;ClOU9K@m<>oH5flgWmaKpPkAYd+es)Ev<4-`r-^46dn=1 z!(&T50A~^*OBND#2h!DA9}#S2W!m6D5W)y$$zdPRS(-*WbPofK!W$JN z8F)FG+JRQ}QiR%n%4LX&S7GQvND3It!S<*=S8Re<`45+?(3Wa9!!ut_Sd&d+-&pU| z#CJ@CIhpiRXRh?IvTKXJBM&1%RNHm2o^4Rl(hJX&pA!$5y(aAHeKE+YYX{!!1zjh} z4h!By#cV(75T_N!^j!;2HS6?U*`0idhNI>;tvTDh><^(jHO-RtjZN@y=f{1YTINCV z=qWaxV*LHr|2`dqKlgxb?{7Z6wvv4O@*T~0Kw+>r(Ex;RDA_Lk7i_%DL{JdC*H16~ z9S+wS60BcFhO*!7M3B+?{Ys0HaQG|Z&{Wq&d%cX zAQFiTR=vV>xuw}T&Se3+-~5XYlV8^p6tzUjoH=!6C8 z5@@}pN$w~FiKXg%smj*0d>xlW9a@mK231zV8xjHo`DHK0l%aje?XOllSjJiMQ# zTjZhUFU^UVP=og}b2e26@$A;54C-Rjq)@6oAP*Znjhh15u3I_~q`9a;ii8MyrACE+ zEEctIoEk=Rc%Q^s*&D?==B8GJmec2|mt?<+f!VAN%H@{Jb98tvgyQE;C6%kWniiI;+^!ePOf4$j z5GijY%U-T47GMJ7qGnwLaTcp|R6YUV?h0e1P zuuO5e*VWGmdXe}agUU)Udyt`G=74L9dcNqmS{)VN@mxvUJC&V!wR>!QxddLp6-Tx5 z%-q;Ue7B<30Wm!qoXa>;0bucu{PCAsaUvEvo3F>nT_#F@I<;mCS!wj!v3l)}OQ4UmM8hC@FCfR`aG`{I2YYJIdP;Q?=C@Y>%=VdQf z3TmKL);lKecrSThsxyqS7uD$0Csf4g7aj=z;L=2UU5gMh{M0LjnQ+3_4Xn^%vu=#_ z_KRB5Z$I-2BeK)Qv4Uc}V!`HOHIn+BY!RG?t>9Z)A|+*GY~+G$s8N5!aeJiItjs%I z9xZJ8^lU>UGH~9CkM4qoo^ROPbhLy%ay}JlUNZiBJYSiBBD=izjOnP-7 z_v*8prkHTaVX53NgHM2C;5H}B)(LvA%@dvAeBbb|(zUYH-2~YMU(uI}05g4m)lR6x z8?Se+;kUCJjvT)PYUjHy3^uqUTgB=a2~^TXf0zrkM*6T^1}l=y(rZ`lyDhZr+E~iE zB|nOJd;f*vM1Hd#4F)DZ0f1laeoPI4&cDPH9Tl?0$`G3l`?99_YbeUV#>_#E-b0|8=Gs=X76;;gCWEG{y7tk)<1i!?%mk#<|@mHd(W99&LxO z=&@HWz9r`GH_#WM!XYeaNLid-pw&uR=j>aKQ5SKJ@{PQ_Voe)w@4C4MPiwZcmHE*z z@!^6MU=PeqU@AvW3wCI4vv4r~_&(<520iMP9`ywZf5~w?AKZYUM`iL}et$W}zLj6{ zA1bPLB}lp7S5^$iIIU-p{E9`mm9rqMZ*KnnCwp*(y?sm!H(0?SFLkmnrKu@IsONhv zgCVUacUn90>)LwLpEDPw8k#47ap)_{S3RC)s&xKRO_3Te*VkQhWeRuW&eK!7QEi^< z)@j+Xpno!{R1+se7Q}};)I(*TaeuUZ{u=$RXZ`Zui{+g=ch$4xaN%wN+z0suT4aA( z4)H))A*dX~a9`nT!I2?t@Z1KDL%}DXZIBj*wHB}KOPIBw)3b=b+3W9hmOM23(whEK z8I6+tMOR%dw!;cv*Jo(qZKf+`*!{t+=p6B3#8EPa?bAGO^Ic@ynMus(mrD2bBQ^n9|`80VK&&2R_8Qd1qQAzflRN+6tRc2mdiO+Yfr+T1yi zH9bzQ_|tSt>3?_fNK4m#Hb)rtQn**3#dH&&2`edcV8c~Hr8WKMpBj!CU&$ivh}Ul{ zptZ2Ag1D({gNoW+PtR(nRKk+XgB#ZNpM36(=94pmt_R*fZ8^B7%y^{}G-93np8ddj zjrwe@fuDDTHRnP$`jS{JH!I^)s)e=P-(Tiv;YD-1B)?&ac+ZAq$M^VpCKe20p?JKC zA7lCXOK}4kzHnq?u#flS|T&{0&Vo@&28$^2Vu~vEcL$6UtJdpO{pARJ7--cuSm_OWEo?Fnn@r zyzOtFqzVDs;2DVU25MR&hf}_w0^0k;7@}#@$9v+h8qAA$glBFHS4DRZL7ENc)wFRV zdiKbrA0}d_P8GkM>ebt*56=6fkKdF(ib7SgOEu!X>9N9eO-gkRcJxFVM8?6Sm*oS; z^mZ+)mfkobbLk~pMdk(Jz5Tk zvb;(aGILmIyuba*`lBmh%7f6%RG^BMdUFx@{BkvVhqx%Z{3j6qhu7FWPoTQnPRHxI z<8;YLCxK{*j@SPMYRHx+fZ^-9pXrj3_-U2ve!Mv7&LzucH{uqz^UGuWzr^95grLv! zcUQ9SKEB5Pm(i(KdVY!Ou0TiRE2NxLJOQG7XkF(9rh7KXmLujx)7}%&-Bl$7nKrs zpfznGpl599efl`~J}Jb zkXx_|LDd!N=5{m-`)Q06=?$Mc)omD9!n1}V4Uh2`hx$E@pc8)-#{{Itn^sC47D+SZ zA~u%WeTfvlHxq=h~m^(M@@tSCIey zs}avHxmheY`UL2H$Y;HlZIUP#9#@$H+7!evpa(~Ooai+svw~@&(5|TYq~1nU)_(KA zx|>`u<^_>Od zFNzmIWn5*lU-`EF7nEOn0-POdc7f;q+T$m#XWl+gR6GH=@?3v$44xM>KU54dIjrez z5?nIxZoArU&s9(lW!hS5dD`v1U1$w^pvb!ad-Wp0MVgrJrw#J%~$JY0;zhxpVz2NgB<@a0sR{G0Bd9=VhH# zkMmO7w`GiZH&c1@qplVu6q>L5{r6b_)mVi`-&%syU0aM&fSZb2cKIFOa@Lv zmEe!HK#w%-&MU}azqEB?)WFukqecYSD$`AGs=mBLpa`<522>`2;@f%C>`RQ%ou@_- zAVufryj-ZPDdijt*VYlA6pYrjlHk6k%T0Rjq4fKcp5M&5I*NZo{ra*s4J*&jqDQU$ zhFIDXau6^TPwg|Wgm#Fw)31avXcacDhh)E-U4~xyDwPtscPbmJ_S)&>xy$=<`>XuH zNjn{mwM#YpTX*2eiPg`%Yn&zxD#dF%^mb0dz?8E@)n96*$nE9ql1|6)6@flpw7aU2 zTVA2%*kCQXCBsY+teN*=@Le@Z>rv;*#k4#WSz%=3RdQ6;dVi6vg>scB(Rx>1a$k1Z z{TorA+&x=jQgp~6x>e*Y%@@@;d|aBTxKXTr)CB z@o8RKlo3U0|HO0R09%9o3wWOZI&Ye@KYr7dq8U1&BW%)Y(WqFYuRah|FS^^H4B3pc z5)hGQ1Psi*)n%;{8<`86KjbI`&=jLcuKlf@O_JbTQ~7DPqv4%`!7AqOM^juYEYL1F zTI3rvf)EetWUC+T#_UV10z}K?KjCtHtxehLAHCnpxRSx#n>kG1!H9;X&NifiY#4%l zbJM)pfV`*39rkSRd)Nt`n=vP8P8qYP&44-BQ~M&8(-v_mzCsn3(^;9DNLCY}67V_L zaKh{j1*jOj#2D%7tFbNj`}sbC`w{b+E(w@MU(M?Yw{&$C)^cDPl-#h+sEk^Q`&^cv z)L7;qP7s&04+l}EyeP~%fPtLBrDEb(eg|i9@2kL?me?PkcgjTQpu}7p9ls%nOdfbD z+Ab5nmVN`zQG`9rK$rz4Ma~?*X|#9iFN8fPaX=MjZdwtJ+%-~r%wtDPl;;FO5HBj zeG#R#)0nzk+W5KnG59^P)40d=9z{d6bQABMfrj#6aII9V^?6{;_JTV!<Aw0hVZfWZ&24H}a$^uZ zq1hRT=GB*vND(mB;E7o{smowz>AqkvyD@WSchy&1X-ZUwE$ut0(hAKR>rFPFQ=Y+a zEh+Uj?+5O{`T#Ni#AgD5sslUcppt}x{U}(mIr+_e^=9%e3skZvY1tW0?_Jv;^Nr9VrIAz(7qPMsB`)7Zo%VxQ0(TT~6-VGqQAgsH`%buu$$HeU6d?3-MC^L#_Q+GJt z$0|KGqUt%Zx#mu_Ibr!; zE}KnrOHwuqsi6~oX%56BJ|lJ8u7<@%zpxG)k|?hympA6ti;NByujr+>#UYDbOSn{O z<}A%+?{8PXV0%d*y)QudT`oX>rz=%lrZ_)eWK$mCG+CNh^cBz90hVyxo@)c+a?-bL zXhTp{R_x6U0-Z<>;6ppfp~Q5<0leYw%twY!7TXu;D+wS0F>!0A2|v```TZpsJ#gJ7 z2DTdJE$F?)g?$7Qt-Zx#B5GKMj=r7&QT3y396qy^L^~3^-A1m1s4@tVySYs$kvC}d|E-_8Md?A@pK_fX@-e^#3*`}2T*R*@l%hNy$S8|>6`ReTZIa4oZtkqNd zZ;2;eWJzWjUpt%KIwWO^mBO1${dJ|1D7tURr&0%0#x>Ur5jku_I>l-sl}#k#D0&js zklLf1KBb^*!qp}xftumbA=ff?=i{RY#oekAPPm$bdiqVYomM}y^bwVZTtsLMUBraH zDWy4FrY?YWHZRYS!us`6-KMy(B^-|=+G3{xgym=@{kJ7DZo@18*VpT^)Q_z$943P5 zaHR8W;S^s8=?!~DnI6#XM+9kgVlDI9_YVJ^e% z;vUi>odW5(a95Qo^Cv*#bG<nDzgIg)Q7_|~xjQY78X@f8`Opo@U!#y3twL+6}1c{A`pPOr}zwI<;j=)UEDGu%@G zA@Hj0BJS#X%Q+kn|w&QFL}exqxZqFetF)fU&tT(yChvVl?A0JOq?0)sAY==R8@g0&NXmF2s_Y=SuAwMaI z6>F3oPw_J0wL!b~BNYPj1h8%U8#X0Eyb$I`*IVP*eJ@!#{y1cL97X%w`IY|261(#g z;3iP_eyMwrR@#oib{%Fem4idA?H<~ggFN6>H>h}h&LpB+b&}nx(%31ef?TI%fV2!u z`~*5z$8KQYMo_n7j@tdO6FCHD@SBvX-?a@IY{#c`^u}2OdH>vvLf7%`U8SxZ0zxhq zGvv>L45JtD*`bT18vXeXqYkZeI$e0NirH@B!ZVpq09_8<4yj0>{Jy`Ww)g&ihDfVA z&r*U3SUj~V^~4(jaiX<6EYAB~yz`y|9z?zN;t62+PizonI#{~e_|dL%A7*sn#L2LR z25anMorW!XC+S96iKNEHxLNM_IGi~c(@hfzyUk4smYwR0CY-CwOJCklLCdR5%Gac2 zxci(1UKIB4-W9{Ic^2ekjW>@29%CDBSJ}dnqRapIOee9$>Cw)o(dyHkCwl&4#&o!@ z2zDCyjZ0c?z|o(fNmW+k+&(&sJkAYM)Sa?)Q{~IiPi(Y$0(h0_aB%Kfn_@&84Mvw8 zOHVm!TD~@lO&1ENKMthmCuTKm*!@U-r|gO6(gj#m0+OrQX2B>Fc?ofFi&GhyCMVv+mR<$7c(u?+Nts7 zaqEfgw87uKw;6R~!SGs*61OB$E@6g^?SHvA4z*@9!^TQ^yq9$}-x84Akx|vhGc*fo zyb#!rX`WGR@oMEFF0;;L&5v}|Z%*QG6R7mto|Be2R(u)=RpOgnVI$iePdmrKd=-c$ z!6#R<4U|%h=68s}ux@HL&s2DgrV)H?tc+cxp!z|WgCqxLU8W%HokE84nM`npLmXW* z3*5DDTGhaUQY1#pE**bMg}y}}Gl^0iDb{ajztH%MM&R>#Z5Y)k(YS~;@NAsuCX9d3 zV=)6;P+zI@e^_f2f8(AYMO_Q4nWdIy_@1?PJ+!1+>nb57I*PkD=GwKXqsrz*wB7uk z*dhC)5C8yBz*P8ukpTcg0Vod5*vETA>Yoc92$W4p*w?g?s`M_aMCYR1KFMSD8cuJb zlt~~Wo5E981f)dJw%>(dvnpj7WTo!EsFQLjOK@5f-et$H6$+P9?TjJU0}bv}3NVu5 zAA_*|!XL`;e`A@2K~=545TX%FhVkA6x&6l%Z9cqs_T#VA*{D*SX9-TN?K~NB0x_HQ zacrosNQh~z*=|X@ys~@y_?0a&Vu9N^B~_c6`p*gL>Ebpu&4nW(ok+)%weKt_W1MyCrX+-Y7lbXR_ecx$9_;ZCjynP4rn;7Eu)Y%>)|q`2Wd|eIrO;(PT4M_vJg@>+M>tK3F_Q zW=ND-eHu*Y+cV6iaE5!ki*0DC(GQvRdVX1_)AfGoT@K9C8cn!}$N;&9Gr_eBd8fxa zD8)^qdVXsc9R=QDmkKkq-(v~)Fz7txe{9`+tD=H_Fh8#28$>^**1=IxttJ$2Jw zx$xDHG5%)4HJ-`m_1LSC{kAKGm%Ps$=j}MRS}$1-wLAD#GLe$djHqH8jt?<8xf+VF zA9GvvbkQcs+gxy2!6yLEs7Z8$=dF%e7q0KrKEcb_#sJx?c0*e}k7n3g>Ncp@L6Dsv zCqvs~T8-1)_i~#4c5UfwvvRK+I_X)yBU2R=2hd-O;p7S6QZ&HZ%&>#e_HO&nhN3J8jF3T4zh6Cu2ef%w|f(lv6DS<2|U=T%^rM zTm%`(n8kP}^h*o9%<0a;F8SaFwlx@v8EqJ`#c32jgc6HL_*K&|vH^gRg5MtiBgq)h zyaK#F7cW93S3=8U22D4?{^BRVN3?|B`+B>}pM5`5{ZAMFyJnv~0bo=UFt#8V1AWwy z*e*61Aa4`^CtL(7Pyz{b(g#@ELKPz-WBqbaawIyHcFz7emAB1Vk*!i3q2n>DPIl|}Q|nrIE-4VfYl}v*V7m7c%>^Dn`dZPZ&UMWv&G}}Vi3sMcK zhozmO7e*S$Gt2eKe-Q`z=i&cvuGRs`p8y!`eNmJUDq(MMzVUHvjw6**2|Yzeb^qpW zANYl@PvrwLSj-I1sXG@*oLv!t!_x+%)ENzIDAtw8#6X?DYjEZ3Tx0ve8}b{2N0vWTgx{?fnqo7W#7PMj zIdzx{(tyMNGF-wvuZA>z)!D@(u{;u>Y6c`N$qNN8LmhnqMZSR>mewYea^i{OoudNrq#1r2AU^DZXWk2e0hp~nLEoh(V3~DlayBSi# z^}*bZQ?>SERcSyRoh$@Uy(&?Wru;m7fKq&`2LQmNtBniosji{RbKafZH_+`U)=1Cw zgV7(E-`QKPB0Hwr>{8;kwiwTo%G7)Di#`~+rO+tm7w|o5-)9tvb{Ro?X*p3 z?zIq3Z*v!wdWKk`1B|A~bhAOf_M_leZ06k2m31Vp9AcS#s=$1KcCwEZuKTmdJZ-6< zJ5|)Deq-*kVHsr7_6sPdFfnUvGkh^fcs35Mtz2oGv$B@6vk6(z5bb1*`&3(njJsn` z5{MCCOzFfUias~fH{&L8Vl#{E%e}6JUv(;&iY0btmwFB=p#3VDKYSZ9OqA9;QR8o0 zRJACZv0tQe(|1kf?`Bhb)%U zj9rD(!4tqK`G%4=WlJ?$<-({8z*^H>yy;OGDU-%)Bp`51p&S2JVxKHYC z5$YGth{w-6%Y*W^FFV;q7&C}neao1%UBfI!QL>k=`B3f7Lc2#;uIIdJ!sessL^{^6 z?D8jlu&7h#+Qseeq4AA4(J4toZTwLwaLus@;A)BmL*EmCTiJL>NKK`&j?2zjM!o3p z&*)2pxO!XVC#t*dxYT3IUkGAhYXyk-D`(@~Hu4wZgYJnHi^6I=iE`fa?nxZT%ax@L zC+SG7C2ruqk1wzi2GPUlUaQ$MUG;fEm*Y9ENP8`6=YtoMlOnvo_*#1zDzqSW@b$z$ zZF5ixGA*i#^cbWfNd$-@fiTe@F-8J*QAqRi%UySa|~Jr-)uB@ib_h}tkW z>l9GmY07>lE>GD8>y=PvdgDZziDOpXz5eoMIlw|5#M0MK8QEuz@_MA{wKC~**kMBV z<)61inu$4%z65-#HJw7jYWy%6=0NwBpZ5Hja+O9cHM*YYHNALP^@QzbA zar7vC)smk1-i63n$5%5dT5z*9auymmm9MB*W^zZL>+HU6bCtAHe_Q9svwK3-lrz3^ z3wdKe3F+tDsntjcD!ox%V+8(@O046UyyFJ`C~XKFAXid+SG?>x&aCs6h{E&_vc^$FHU&I5~!R1KCV?Xf%gaLR|xnNVeVAk<|vq-ZXk@x46c%)~nXQ}~Cr4R*+K&=So)>&v%N&K=QiY1{TNP-maY)*fyg~$k$uQ< z2Pc*6*>so%R2?^c;)=}Ar2ZmNO*?~;3dZY+aKqC${m>Qm>k=bjkqgtU0uu@VU~8OyU;zNQ0sl4eVZM3Z z$OYnHC>~7ZGPj#jPSY&yu%S;C4m_Gbz1$IwtsSj+=ax0&^ZCa=F_VQFMv8A$yQ%Gq znHFS*-2h91Osi^ifxsYl@nxz@rTX8>P0yP0Rap(^X4CeTQvcYmC5KtDv1W={$%=6_ zn^$rvAFYyqkMImV0j@Ol>(lDfK(=Pl?k#+EC=xO?fb!-tnF0Wd1Ayxonr0BbT#jW` zS4>W6+2MsRAgVSJV4appA*wc@B3@);y^t>ea&d;Sh0g+X@}M8sHwa^b-Uz7j>#tN9 zg3Q<^(`*DAKU#6*+IHB%f_Zx8Kj+()P$C=`WK_nh%VO6XRWeuHN{CE8>AkgKyH$Cm zPrV*@XJIHNjN-i@G^EWwq8J-hXIsr**!ySR{I_fcI0$sk@W7~I%WD(UNY&*@my4M_ zOA;lBGO@RckWB0PhPE8ZUjHHpdH137{)15xoQ?q_$4eQ(hFuKYhY2eV#1LNc)D!5= zr03q<*@z58FUZ!JVGhI($l_NHIqvLl>;zKk=JGat1H~NbC)-Dxq_A7?)#;`T)dzxPC63=jXRA)MQXW)=$`1HsB8-=R@XjNN4OZ)l0*J-oigZ- z97AB1te|u-S8;LC#q7>Q-H;2RVfxOP-$sFFL^pk;+~EC}nZFUokuxZW%fQc0(` zDfcA(k!1!Q(TVgCwODkL{E`}zxU^`ZepXFg3dBg|>t2hU1U1LSkT~kyXxCiU8ZsC; znpJ%#$E`HHJ{B0mZlNm2MK&y;i;v(P=Ijiw*ZzTT&T|Pa2v`KKs^OMlMif*tSi3(sVpZmW!CW!hnVkcd5k4)^x2&Jn^2c2_8HF?rD-4Z zQ~%;87;;kFzEIni!5-A}>^zQrBNoR^>Lti-$!5FcSUHDae-rCaz zeNvt3$(s2EB^0HCx7AkJrb>4?a9tu47540dD{eB{L9VzucNx_VZ7URk0lP<;>aD5P zhN|4Zg90pFibB#We=+J)Fk;f*QC4l*yzOmQ96M15+o57ZvD8RD^$8UH6096rM}|92 zmEyBaiqqM@d2jVKaUHc9rtTj3Rg>+m>-B~rd3)h}vi=L-C?9)%I0P;!YE;ZNOGNg& z&VH9755;LVF~W!R-9Y*J40og*c1Tj_M^KXnWJJItcFY-(>1w>niY-&J{%IDh;lFw8 zmWn_k47%boiDH~AJl@c1o*YIn8q0^=83@%7sWo70YPzfCvb#)mM77K6MyM}BD`~Vo0mNGM z)Wn4a8*7jHq&529j@)SP(@@kCrD#qIkULFB+uTrOi+H~4k^06v6s{zu6j190G#%W^ z_Abr@6WA1uB>J#sW?{uK=XCT%xCaS&#d#;{dp=vQ9xtjCj7q6AIkRloZ-4~**ivgocINT?KoiH+ zOR(vDGcwM57)D~_-BHjjL z>D`*y5%u*sB^aEn_NGk6Xt? zhj;G!vFpG@zJ2}(0&^$>4HMp=G2{nE=lbiXkEsrjjBOm40qqb>u~276*>rH;+wzVa zAx%3%Yf_FejUnObL7tP}89Gbev*oTv76*pulhpa&{*c+}z1=zEyCb1Gd#g4g3U>kJzz9ob<6tj+jNGc0uX$g=rTe|;2W81DOc5Xa;Fs;5Q|j{5}2 zM=%bWQx}!oa&@yOO3n-Gl10-aP;cmkNs=YfEPRir8A``O} z=h|m6%^X@%Lp*#+Jhpo=Y~XWo3Zie5+UDroRP4ZtBVz+=Yw^X7O_g6~iDJ4d@)<&s z?-dz#&xaF_7;NFPL)hQA96EZ>T@0hRxAI8Y{q}{1O9W`vl5~H)5bB6|#>Ij`&^-po z8<4~&w=ReZqQu*d-`Z(bl(b!$y&WL$7cQLYAPnuN;Nn9Du~JZQ4YI`I^U`n9#6eyN zHR;^&PO*Aj#=Jvj=4geF4XUo0;x`1moX*j6qCM8SyD<#R*uL1znb?e7&|mj4(HlhT z9OeG;IrVm22+CmA5l;e&5Fjfl71~tQ!evibFC6t^zag}dMOD_{_i?+7;>_Q?Eq?{d zDh?6z(d9{h_NjzFc!86bO0@p%dmY(CAAQ>AZY2l=ySY@`0M7$?n9M(izIx~qhdMmf zAgXHX;_+T@?j%?iRT*GYws<)y4(o^S8(*-Y{h;}#a0dI^c+p_~?q#+LZAx4UYdbg_ zcJdYm#j71Wu5g6`KyfNf6zu|<6i*R3Ll}<-NCsfwKLhJo^{d*KIXH-Gn6GwJTJtP} z&1uTG*f!!Si?(TXW6^;Lv=b(TVF4PS4-`@pzX(i>9)QPAfBrqpr^)>8*=Try4!&JP z!!NY62RzGlyTawC9{=c=3?0fSH-9taw)K|mdp}Jvk7mhQBBora-0c4iHC9G`AHE z1@ud!q)g~_w+TwFWUz%@T+ur$4xrWj+QpsD{nbPemI}q~mAMo31b<|(TU2knO@&Z` z1@VrpI6Me-LlSFh(1lY=TUY!4Rz9Qd!3lZx7zjU3IL*+#Q1Lk)LRW5bi6@mNRhs1^B4WUvXwx zZazF(ye8)$LOBg@^A1&-0dKp;q1(vkma`oon?s+xB&&0(CGWdTYH@jI*NZ%v#<2_# zoq%Np>$#)Z%Awn2tP#_zu`l8Tx8*$asiQ)N)0#iW<8V3mhHW&fOk6Wd+P*02i_gfcc*l^8q_${=X(?005w!#)S~M7Qs>TD$zr| zy*?lyFLzZ;(D%VGfRb)& zGgO5~{fCI7A-A1De%-Sv)JMO1LAi>0LTV&g!kM7fiOTY~8$P}&IeRmE^~+){nYfbo zQ{uZjN2B(l5YQf3&t*?d0D^l99+GC>LLQ-<-6eWmdbvnb#JniN#D_k5gi$r6sK3)W zR||Tt(9@N>GoTkz#X+9+*z3wv&Kj$F<%U1}<2aJJq9Qz&yv+4XX=aHp1LWJh1_nLa zsBMO-I8ii9+vh^#Y)K&kHwNGBV?tUGhBFFez=gWML-IbX9Wt`cHV)Shs?A7Dv;ArQiPc zG~$rI7a7PNgh#$5pPP}%cC`V2C&=#W6rNGnsXqUPm_yDQV)e4j8fD3dZj;o07wb7bp4OGpYt|`mhAF2jbsOdyp`)GzXXa9&343=H0%M#e#l22bzOU;w z37`zjLz9u?a;LW4i-Z34-+FOHlwWh#1my2-kI4QKMZ$8tydY~g;yDS^fg)Zj6Toh- z)jbLinqOp>tMNsXQh^Cc$C5`+0MbKKpz=MvZeiCu;rC(@<02LEmFzct&l;lS!|Hf^ zL;etW{k4(n`1~wnVN=hto>9#MZRdn|1ePW~{2u@_K+L~&#yz6c?%7B6GI$<^u}ei) z+@hn`__>x!`U;CW>sPq7I!yz^VsEfbs&Oe(8^v^iy#ohaLnP@Ww}C0FQr1bqv)Pk; zS_W6p6~28N%}X27SziK;OD44hICghZm;I8z)fPjaYrD;c^qNz^qcxq{5?6?78lm}7 zZa%2LuCc_YNTEBprEx~pO>bvZ#JXNH1w)L-8KyG<#kIs+#II*e)hEUh=)qpaC1>T| z;83`TX0HvgZwl3{aZ<#Ve}P77ahdO3yipF6Z4FmS$@8idrlwZ>6Pw;J*L1T`*L1X} za+GUBNhSCj-YJH^4Gl-ZV|gh~W~(K_Y+{I5W}DoN?>xl|>Z#4d)g6bw{2l4KDCCmd zl^*=v3K7QD(WeD{Iv<+4D-j{|iP`s@7B}jL>J~(8shA3xaU0?QUf&P|-ro=c_V|(j zy}l&l0xv>a-_Qc&9vG2SARtuK46I)tT`L&iRezIeH-R60EZ#8zrW8s+7YqQ+;?6F77v z0<{2Eq<}yHP%}uV!n9=x4ebqhu%XOzLo}*)ly0I}JiVPZDUHhGjTrb;ymrwLl$uy; z%)i;?q}qjy%aUS%sU2i3?(WHBUP*HDPCRBM(Nub>wCPLY*7H-V=|9~hgv>8jc4+Mx zLKfkVAHX>9vt1RAj>zvr$Dq@i8t&5fdRon(BD6gL4xK9=u2U55jilo)LD<5&84@iO z#+=tlAu}XYIm;|D+&WT9Sykj8Dq|T{*1lsH2)}U&+OJTe&sT6%!l9 zAoOsb3SBKkIc}T1YN}nC(Dk&H-q%uc>-<|KOeU{xWIqyyo#o979JQx94Q%44KZz2* z)dh{>VwkHMU7PNAw-<&f60(|~^O0(#QC}M3mt#_`&-X6H+@HDt>EFv|uTn4N7ePNv z!a0<#KP-#?0Mpq$$@b~k)2CyS_}R7v^p_a9C_}nQFAaHYq)^<$TcMZt+OVMFzpm8H zOXwKFwP|E-kjlqp6|3}9u&lL&U7%SqmM?1c^&e9_!oX^1nUtKm!XYmtZy1AB>uZma zV{eIsPNpF!)kc4VmF)*|4R1ursKsi&jW$ExIs|hz-J|r6Q*$E z!by}&{0zTw^>JEmFSbbBwbG1BLf0yiInu198@xdUc#4}hF~^0(}CP}2LPFViNXZCxUN)k=jT zyfJzcuH>jGY8g__5lYptbWPg|a`_G5B;)@8){O{-$cEaZ>e5{-B#PV}1Qa)-Ik6P6 z8=Xv<&jiDh9NdOl)%J%LH#2pGy-)EPZcL4|Db4MTXDMPHS#o3=sKr$+N7ThDN`Q=- znl-8Z*jI(%5NtJtiJ7kk+AC0-&nF9!0{B95aH+5BDvkQD3rSOsNLgoTV!d{51anS0 zrtrA~sV4`$O&Ia2S!}+XZ8;>JRMoMPg^VHE)9D+AVjtaJo5ED%Ik}z!Q3~NU{$}7Q zE|%d38CP5OhiOOF5K6-h?GWeXjY7gRjDrDWgo&yIEVDr9>x?Zw=>Sn zOs#S^gQ-MaN*SI0=tQw@@<=u+hLSZ*ZZ2$-E}m4G$x?xjA;{Hpa2Gi})n62x5&>cv zL`$sxHLLPAGFiezR}EenZ-JVb5EUo@000IXMx;e*%CX{FDYR9*LPhZs$2qpw0Bil( z)q`qM!sMBq$x>3IaNZ!kA}yCzoP?=N#oWBH zVgdp|9G#rL#!k;*Mn#a@1)uNpB9YWjFJy-~GUN=DrXx73%B?qzo$jiu>FDk0*Q`*j zDT9{jm`Zn36rKX66vl15juh4a$1T!JYM3RXDP+>tuF^`#TF4<7Xu_)Lo7>meyE3M} zk$fMdcwBudxgPJLa3>|1Nu{Z-E-dPzn0=8R=ZaLXcx52!bZxXk>}o1qMN*`DMK~yx zKrt}@O#l!_i@*X%4Aztvs2D8;>IB|pKo^Pujzmz6-{=YoAptg^q5z@{h#1%eGKW3V zlS^b#Ft))}CKVkO5X&rOfy)N+ZC(_b=Re3G_+Ln-7A$o`>zFLELxPSr81t@L9)trD zpwiSmaavRGs3iqd%g2*XMYF|HCkwzZ{Skx1{@L*CPF7v-s`E3L-OzU37GmUQY4tCP zX(d!H$;U<F|@^S z*~DK)E??l6fBTb|?Ag!zc!^%?N`|M5E2lI~6L5hBkJKC7@=;z2Oj8kqJobQW*$ry_?D9yeEhs5o_yIsf&b{ zn|h37G)tGx+#2FYppleR=kZD$L^F#qF35L9dLkVuN^V@IEcPWD({zPGSZ1d!^dMUr z``14}*NYiEay)GIKIxZmFy*hu^O0+`P1O?)V$DNX>;WjMT`dxop>Dcdz8a+!vGvTp zPXvPH<4!)VG%;i4B&SzN0*fPd4e9AtCpGUYS-G|S1QUqd-iTxg5@ZKB$Ak4S}E!Fb|OW%&d6-(4o~kUKc-`b zWvL+(z9CQkP?m($#ynay)Mxc7YB?kA5ewrg6~n@ycc@=-*0)_%^|K>VbR%S`3Q;29B*~T3l~pec#TK>_soTRkB?d_nG|!~jom-|KN)I-&zq%=X5XGOhrV;AL#C#pm5FA=rPXAO6?bxgA#hme)P zh4G!b+Z!La?LwzZ#FlaOgM(6+VRFtd&39DW<23VP+C-9t{IODB(+}_Q1N-`<9 zrE^WZPwNcsQxwE!+y4LtXd%WzJv7CRP0#!ww?&8!X2lyPE7{&a7707+xdGFR9bO>Dl>d2;^% z;33X2Sc$0X&B@!HV-sd;Znad((grl9bsmeo604NIhcP*!s^wM8`L{|sO!%>TK26nf z7^ z$=NDr8!D)WspG?IvKg}c*|r8eR3of3t!qwCKR4QO5106+=P;frk6MS`s9s5Bc9!U+ zI#zp87`_Bz;~pssh`>y*D_huwvAVmO*>zNJ78!r4977evrPKA7R=b)l;pg1;TTvI| z7#%N6PZZW%YD>pu{fmeBE8E#hTv`Q0zBSSX^wkr^Beo>}04hWLmF=XOZlI$OFDZHE z33_Xcc!n#E%6=k*H#OEJHOU&jgN^W9)eJvTQ->5LIV|jv7wD&qd?oz~c_%)j}J zhV)7+jk*2}0^fCSc4}V{`-OcJ1o7M#5Q@B-c^1j6nFS0zDKe86-tDyGIaSP6}Qbf@KL1xv!RC( z<}RC_UQGPhB^S9}zE8I5n?!x_5iHU#oZ#WaahoFI4(8H5oyrM)>rPY1N-?6At8piV zrS~m2`<9KgH{TyH)el7jg%qu2@iqNH3g2_XbL7a{!lxZ}c+1s>FQR~$i;)kN#1)rB zi$wnb2VmK)C1h4TX63hfhRl+sACUvNK^31W^>UQjDHOQ81xzVrRQXmf9~8!B)<#|} zTV7~~Iof{dnhMkA`*~2jGN$y`8E{ayvtQoj z+!u6Hw7ts6tjNFlX(M`O6Tu_1SG`n}+oWq7#yoW|LdX0?Cd2j88E}jngCg@)`j7IY zwnVfV&&DC~5KI#E5DpP)U~FFf)PI#7x)Xt07S@ytTGgsJF|lnHr8kW%zt$6B zb7UHY3##(AfA@W7G{45bRNk3}D8nR>X=zdW#m#+J>zl?s;c76kF!p#JH_mUqF`;); zhu}Loe;5ol?9);qFX=xw=;s9nn>4`^9m}8d7hWeHeNwj;Jf5>ukMG``8zb77MkACSCd@6YDWJ6*h6`r zJO2PpWk2g5FyzSNF>)VoYSGG0i-#@r(sFY6rx~^MEr&YhDR54sl}qr~ykZT8#Y3qJ z7FL%eYTgcX+vOBq*_1D))y!N$ln?aW&|HQN==bmuRSa1+VHd z__FQ;t2Ij3qq3{sRikVcvX=(Zs$ZQK6fPdc=7tN=sYU%oX>G#RiNdaA4e^&%^u}FD z6Omf--++1k}RyPjsYYW;6@a}dAAQ|uJz_DZGn6&leb)vc+)#rioG6pPmGS6lnb*ay9+uA+225(e%}@8erZl$BB9>j*nAS?>|GgqF918 zKfb?0oYlRL-TP~Elz+*nUzAT%Qdgq*f`87xLXUp-CV&0_l%6e2@L^CLJV|IFfxpzZSruf!R67X>MZu(=MQWS+RV z)XmlK$}5MGpD5^tGZzn^SvMy&IScFoPdEM&x7$1^eeB_N{R9TW{13JD`L%j6jRvv1 zA8uanx;8fVB?@u+qbroLV`8hy4&b&`VCDO&-=7=e==UG4s9Y`n32GxH2+XZLfj>xJWJx*pqHSKP=7~Cr+I62x(eIhnAoN9Fb$n@;A6O_}-j2#NwY;w@XZN%E&S`&)l2qQAgcha;DBj}Ev!^TG zI<8ez-VuP!Fqt&&JWcHZ35G)OLI7BpbT0%epr@s2nPzj)Pt{`~9;p{!tfgCAT$liA75xa9m&qNg$xLd77=BGF0d;f%H{PQXFZ{K0k7NB7V9<=h&Q^pgWQ4VN7u)(do?T`DkbGj*QLvS9u2F7;u$U4aUBxP=b0z zsI@IMbvVwjrr%S~&viGcTq1?xps6&HRK5`pq#%-(g_(ad0+}ZfDpL{XJ2@na(h0`# zO~N1pAOHXW000000g3}&+r%5mn=zJJ84{XWB!rh?{b8x`NGw2QRDDLyt)^mXMhCv$ zRgzAU`D6txQt&HMr2``(&s9aEHwx+<*SWqx=pVtp1eTf(ZE2WxYdt^P&F`{#;J8ag zP%=|40;K>9Zqq3AYx)MDqaAQ%m@3?c$luMMwu0-ZxwUt}F#%Y|5v*~FCa2PQ!4RufY zq~nf^2nA6cQw48NJehd4ogn<{zK}T|)>4J!ANZ+!e>8)3EBGmqrq)$W-5$Xh=BTr3 ziAtJ9AqrCQNI}Mp5n`@p>ls3zR)}9xabh5;TN@VUMN5UzPmW=Y%Bw^$RwOj4lT?*# zRS|;&2O6x^O(0LTV%vMVzm+k+1d({=IPFQ{j+xBoOineq9@K4{JW6P_<)b`GR;W2n zL<(QHqDjp)FQ%^y!`aEn@Xt9KX7eViCzw<4?j)pE!fM?ks^G(V=HBPx<^jYBNn&EXqiLd6jyP2p&yREeaMFO|pe z0Fz=T!YKgBORyCv0003fmzeS}S|$5pPFe)3rIJCLdCR9%#=27BHAts8Dz+L`OIueZ z5Y@ZY{{Sj`L}OOWqGH*N9;y3Mk&FB~?~bp<6nRWz7i-$6)2kyQtaViBBvM5L(?t|Y zRLD{gg)bC?=aJ&^9IAR&Hq@c&HPtAFNWPJ)w@cS-?B2p_3Z27}prK=xYS+S5T@sH& zfWUlMtwc`rIryYhWHqVsqRp)>8eWoX6zd$KCpVg`~7xbEi6pjS?f5~5w?bzJVuivyZ4dSy`>Woa6 zuJ@GPqp%l>cPmM&8eZu%;tKi>f}ay4k2>GdRAsaK7t52lTd@i8^y8xX7sN!QIBKe9 z$%yeHC0c7pCQRexsyP>DneHzv#EM+efsa0VIe4!Nk*qJu3x@!>aa#DSbWQ-Fyl-`;>%SWhF4i`F{Wzu zO)Fs)>_mb`(>!v|(qAebDyd!`j^$2AnUd*T)S7T-o6Env)KhzzYbsUoqiU+Ei-cDp zFJ(>i+DRVhm&C6#(#T^H_^MqEb1jCk7O8u^ui5c@S!1i{Bn(QM&zy-{T$vt5vZ_#* zvx{(|i{h3^UqPvaD_6v&^Dz_jo~hZTmu2=}TeMf^As?&hq)|4i5ZTD7$)v`!>lbgs zw&w@%Ql^*vbxI@BFzZu(&vVCmmKui&(&}|SE|*L7o(5!fsoMVlwtFQJx!o^ysB6bQ zEL$`^CC)M8ys;zB_#Vbbb8 z#UpTK)fDSta&L(+l~W_6*b+9Gjb=_Gau0@1I{AfD?lMrVmqae(RNep%dZ%x8p}%L_ z)AWbvycx;E$oQTi!{TSmxelBsdTME^!L)WV=^wBr@Yg})qp_4l6=KBir=lg#M%5L z2MEi~(@cu-Qg{R*3Mjp5c`QzCty?hVRd(NE)m3h@vim%eRXYf>_!QR3proqpFzJX^>6$>#50_2Cut+YhkH+2*=GAjO9v#t{ zEG4!z%D+<7UD5WmfiUCdW0_>+w1i@Te8dVYV>`~5WlQsVEySTNG%S|S}EXB&{8WQ#3#ZWD=7)Z$&##a!hprS`ZYtJTVROhcsnMC$l*T;<5Q z)*Gs+D>jExh1}Cir4rLFinERqyy1Tcr9Y%$n_=C^qYdcH<{<8|io2h2%rkWanv>vzC>CBAhrWL?}gwv^N?3rMa1aF{6}_1sqMM{GvG- z+XXq2iQ>?j!9j3mau`_%mx3no{j??umHz;3dx+v?{9-CBXMqk-R;Ft8Bn}-+7OF@8 z01-Dgz|NKZBzq;v_rYAx0mD`OJCW?={9_Ioj|2w{juH^WRJ?4z+96>e;lomFb^h|_ z7%K@$Xi8id)U^~@@5fEOY;IBunIPemtel&>jD@YV+S5HY){#>s;>$P=8Cv0T@8pI0 zH+1d3$7TCg)27ey?lJ!WOL#?!Yp+M-)p>5Ec{f?xpW&+H9sG1VHT^V%TBR<)B@_Pu zs&w||veg!e8oi`i%^yRxrW&MkpzYN@-33=Uc*R$-qEd>H&Q+qHJ?UKQB$Y7PXDY;Z ze?PJs{@M3%FDEMQm7pm&7u3^!<$76$K67`Cgl!TOPd;Zuqn!tbCkZBIomh*Rr&Zl0 zSQ$epA4$-RKdVx#KV2bf8$;Kw1p>5HU#SCbsJ3DkxlYC)U&uhPOVB_|i`&upMrOZt zA^uf%_EPsM`9!`gstI}t3~e*)Sue#>h1Tgyb#!ezP^gRXZ_byX7KFiMlm7s!TAW*T zvAX-J)M}-CX_fx4!qhgC41k}YqH$_gmcD|u^+ch3DK)~xB{1j-`U^9vhHdVf$#={A zV}qDyX)Tq9;Drt=o|0pHu(a50oIimsGW_TM<1QK*a3K-Au3iecLxzKc2#04c;}hta z^>|~mm+^<9I=nC<#3d!#fGk49V^EvY8F3m*lAXk*@xzGH$Z;v$M~X0_1Z!)|m?(yK zF*2#t`;Ip~p;$J%POMJ8f%E zsxuPuaB+&9rsS%xav-WN!1+J=mpZVeEk!Yn+BHGyD|wedsub$x8A3^C~5% zb$`>@TQJ7P?^8z?+7==7Mz8k}h|1GCHl#`>g^1RX7}}zV+yaV%kvo+2Db8D`A;pOI zr+Ku=xQy>B8_deJyK%|BR*wyb zbc&G44VKLvm^jUOsc=S7g7)L39e99E2KkW>5ujx$J1Uh!}!;YegcjuyiMHF6lyz+kTwe~D7nBdXyvNyj+| zUzRJHi|8J6q{|JMWJSBU?M>GT_O8ZLW6~sAHQ6bR`h$Zkw?scK{{W|Oq0Mxa{+D^g zL(rn3`uVq7h)5MfE*h!kWD!?{LYQKVMJsd`FX<`Njcvry8qr`PsjTT{ujwhw!EPXq z8K~)Yr|Buo-N|@Jm=+nV=n16o8ulpRGsFdhhR+Zp(*;;7ktm84EI29#Hj!*C1c5-U zgP;s8DdEBJBnw9l2Z1PBVz6L%0JN!K!A{~s{qc&SimiK-=twJ6D_C{zPoNUSFj-hs zul|UaLg=m{8nGo{#zktPeGc&4$0*X*_dQBkyt3Dnk4eNbyrQJ3Z#Ps@R1Yl8_h zL>Q<1VZ%X4N!%}m480WYCBKu9CWDHJ+%J$t0xUHBBm#qt4L?X2(2Ay$%kW`AUw|=2 z7Zj{DL&z>FTaX0|Eny+FLW0&31qy$(q@hABgrHVpuY<_(0=3|%0;Ot_Y5xEy6}sXg zRS=f3&4EXAGrJeH$?F-dn$=NvjMvNy3d+b=D;>6Q{WN$^Q03g`QW!(@PnvXF8;O9;c3k4~UjE~V8N8K)mSa%U@+-zo zMJh$vy+qelK~fHM{v6qqRX)onI*Sn+oRs?oxTAJ8Kf0Ec`yN^@L|IxCN*9+;FR_wA zE|h@zNYoeRd5v#)Z zp5&>#UJ1kk@B{z=2><{901QwbZiqvoVjP!?aQBPjix3)?#@!y2QlhfwNB;n;FV3qE z3NhA_TL;A`DQcT(y6CQ3LW^S;y zxFF|6X|~6f*9086jm+F*&R@2T(e))YS~#XC)LE4n!`*bc>0;um#~|E;CXX`W7_46F zdhrJZJfu8R=KOM$hd|*Q3qi)AzbwwxX^T*gQOq&SgU0^=bu?GqQoqOW$~jXqTr+L4 zDrJtGVHCtC2x&2tQp8G_tx;1((vfORN*u=1!a*|WD!QlOBvSFrakT~pbt#DJOkM^i zV>)?O)ec)BQ95dKwLr$alsh7QUOBkO9R{K|ky1LQr(1Mp^cGSc&fr<9%!-`r$Vgr? z@0C9dQ(PMzTdW~lADO)@hSgcxtQyY<&oO++lhtEsrhhrU`sYpxgL5fcRfs~Y+Q-Rt&r=? z=i1?gXKxikaP*|dYABk0`SbZi^qfe&OYlf|DRP8*!QP3sDS8enl7UVJo@`bqqrow| zsm!VLj;Z-(FoPnwj2vw|%{bWPbZ?ziR5!Y+sq~Jiydwc6V^a4*fs&U1Do_9bUI?szYr^M7d6TvHA|GwV0Qa72+n3qUC2E&7~WrRp#+k zypf%95GOP|UN4L+6;wq#vQl>?ROwc&TQX{kHf@N5bcjNvnq|6D5TnG#j_A3m=o-Xk zN3hmLmvaKxY~rmDq3dR6%irAu-4>0jRqgC{jem&)0KNCimw>y0_D+Q0DH@JcJU>XW z6^4qHIk^&|kxD{%gYNS1k`pM0%;q9-ADK9qZdGwyM^Pv?1gaR@@Bu+>JAlM--o3yG zS2=r%IcKG38nd(ZbO?LCIfjNQ85s#p`36Ia5fX=_h3bCKo5tk(rS^}@IQhc0V@5rE zMx-cp?j~Z_D;9~jH?~BGQS_`nSs4uv5j5t`Jzoh6)FnwZO~@lH1uEr(v=4nHkyOc; zgKLJ0<3c2itm{@3W}(tF#_DQXzlu(a-5W|HFw=mP!UQLBpwsS>N;kEN9o7`fsf_yjkrP3az!HV^hEl8)ue?uRx%k9`>BMsM4y zf~gw`RMaig6vZ1z)Kk~MjRrCzw=`0hvzPJZ6Ji@G$YlP<;ui3TMnAbmF}Yv+iQu54 zwGhQOEW;`{z*WTA!=8A-MtBcNdxmzmJYi1=jshomE2W9Y4iOepyqh{&_Y6-33=$KQ zxMxcT1q|q^+*@5^OHV|@^3}yLM~CTbZVMF<^qn1mk9|%2i3pKXM4Uq0*dH8-w>?~^ z)L|{HL?;y&00%u&pTNj`uSkA~z`xr*DTybhT!&enx}5hSS_VN82LJ^SAMcEbCcXls zqYwa80RROEWXf#0GUP9`3W?rii3y+XW~9X-&SIvSJ2^#bk&Vc{y3q~iW}B01y1|zD z>eZUj^tlU)tcdp~>&5s=sYvOuVE zxxTWSddK%L+C8rruiG>2L>J*@o|5WU3&}4WVtMRSq>{LLSkFn{18=&t9nV{rkUVvo zM<-j>geBD`9O*YfU*r&tRa1isQxoYT-4?88ckpw4q+11dYcid*tfy6XcJ=~16wtlI zB3uFs(TB2;z@Xu&R;FeYE<`$yeNxteBobc&M*smh0t$bC5K16W2{QmF21Rrv0YY#T z0!09WAOfHPAOHp^Ps0Ml04RijKfyimE_xD(j7y#C9Wl9ca}livE=xnS6!J&Wk++rZ zrozu|dMOuu2^ven8JUc}(SQ&F3BVHHAjXB2 zQG5CVf~rguG9@pz6a>MUl}UIZn5i&=5CLG}n7{x41)u>ohw_P4D{NQ6%@EqVo$N+d z+$(fq6U9Out;Z6td4#^PxT2>x^4%=P;Na5h9gFF&anM_@ZMH6lfN#0>%(4}t4I7mS zL$>?^T@=1I99E5~<$4p*_v0_^Ilg@JIKaNpxe)r(D$JY73C2d+04zdqG5|q1%!(3R zzr@i3Ekaz>h8Y+vtw~JO`d0w0WY04sv3NuiQ(a}1r=RKxsamm6^c5hgp&+#hy-CI< zbSD8o1d+?YFcO47u?fI0AYXG5<2E|7)H~mBEjpAnnbbAI!eREw_7(4YwM*9;tIYTp zPv-i>`z6}7%;GxMpjxV{=Bj5LDOc_uUGdetHW2$iM&-J+yIb%=&Lr$2TN0oPiGYv< z03`MUAOHa<1eT(yFt(<5t50_CT%qnxOkb8I9KI&(8MimQxHqX~QsJDc_{AFynxVG~ z7;-6aPO2u3aM<4@Y_@FH? zs~2t~FV)<+w9q4H`tx~%{qQ0D$p$8B0IUTWQTy_te7{yfuNI7_f znoTz(K9>e*>a{+(?m7L?No&7muzrEtYo=^4y;Xnh02PGRnsa^!Yl#Ym`h zl|^TVNXB^sc*N#DsN=e=Hro{p8Dx%2$f`@cQxw9w)U_UpQmpkBuA8$|mrCPBBY0%u zh@qV-RRsyMpg;fs0000bF>jOst!pS}F||Z}Z5oM;>6P{m(-f^;}?z$!!0+vbBX9^VQEK?TX0~@U%R;cOqnX*`^NHu35ldM+}zE^Df3>q&UfWdXScDtXyX?5#+Y;=Aq9wB`rsAy-)^3%;td^CTMAaQY zlBaJ62^-x~jwd>*yabg$9OmGS)~YpztyM{FLYJxGES98_#xv@IV+*z_p}K({1Z`#Dv6V96)tzN3#CaN2aaB-*H$EmpqE#`FO zTp`^Q&mJEEsV|u3H-I?k$2H&DGKlnI7P+QXJ)oixd{-UP?&~c+nw1 zN~)GagtG=W2WQnKecM!rOJEAGvi{q=?aevisS1M6|FmxvR3A%(kVz= zq$w_xgT_8(@tv97hT9T^;#$>jGd^G)SLA|-hcs@{x396?BfJV$O6^flC3N?AY}s+u z@C;$<@GL!HdQL-;JsRYqA1fW8vl^x9Dwo-GQmcvPn0@=9kx`jY`OZ4ptxB8sIOFKl zJf#mzQ;4*K(>mplW5F?0Ce9HMu&7|Dr=dmOL@@k-1{hWiJ+0O z*ZDjV<8k@4V-=`+93oohTzb?_i8EFv%_IG*6kEo-&W=w<<<>Myl<4DADGD8^Rga^h zQ7CxN@C4CFA;A!2ktmc?E}(5XxU*SO%Td$oN3F&wP28053{zeq#x(CVCh&+8?No9h z(G4wn!Z$^?N)p^Dv}wAM=@LSqWulRf;riE#Do3qP9Q-#Dz!!;zEm_{2&1Y*nlI!a} zZfb^}xl~hUE-55qbc|HFMoeDVnPzwJu9M47{{U`nEPlp{!#Q&0$b6Y{WfvmXk?NJ| zzwE9=I~jW&k!mGEEIMV(j~D?c004&!0TC1{0u|X7A}L#EEc;_#Eac?8R!I{`e3Pa# z2=~zschRUeo1}c|B2}ln0m7mga{{%aMYTEdo&E)F^8F$DCj$QfZ1|od>z4u7Z?2|; zy-+1cMF33{42S>##0#eaKmqOhLKSF2swXpW*;eKoOSL4+Fjj?zr}#?CNOvSIag1Iu z#S0YR&tvkqdRos(#?dWJxyN&cObRA6wuM)uU*Si;aiFPrp-RN5=idVp=@S0{acpP) zjW21o-)Oe5^E#_%RZ$We@(h#iXeHt#E3HoSjtwm*lr*aNm>yiYGX3Fd9dAw#K198o znLVt6Nk%>)4{S-2CkvS%5l{q-yb}SDAQAuoAOHXW8Gscd*vt5Ui<87$%a_{7V5$*w zd(_Nq{>ba=QIt+1lYm|BT%PKs%c-GzjAad?K~lL>k{s+u9h1VFRdRjKk!fWX4Y`@2 zX03)zDz>V~TvQ<27V-o?hF`29R-Dfw>{5563v8}U{N|>ewoTObfoka*9TWD3p5Uhw z8heB>KPqlGy@}#_EGMHravJh^h^%ZZe3HK>Qs z;j+f$KxT-PD{CDQ#KN;8KmZ8<2%E$cLmH3@3X_El?g2qkTi1ADLn^4fq!bk#q6*BA zn7nJiT8a<%zyJUXP@Dxz)fY!iSG6>KCr>oaU&$&&lT5ymyb*()L!XLbkfdduF#bn) z*0c1k;@cB%Nz)1>;;on_g4|^l;Hu=_0eFQKRa24i$r(wGmtrd#H~VP6FYs^-i|0~L zmXeTI%eru>)m3vE@r+||%q>QraFmltW+b$}Nkhn3h;$mM5Bn?3m_d?r0A_^XNFV}Y zkY;B@Cjw}IWLS?t2596`afL#|M-r^v69r1vRTPxW2!08{3LPeRnUclgUKXLQ=~^Cl zZVXE8RPH4KMJEEXU?l*O!f^!Bp)oGxQ#j?9fkZxs_DN(Li90$;uB|`vv0wdMajK|( zd2%i-(vBMDh^NmnvWMtdXCxM*P=S4*H{C@_kpns?089h`1myd`XGJFf0F(e&gu^P3 z>Zv$XvKuM_!cnMzO{!*&ra%&?R>X+atxrOn=@_Re%u{$G#EPvqSv3)<%&c!WD&K(S z_lzxU%9{}JIE-A>u5VO1{xsNz@kYyL4x}!SHHk$o4eD6ccz22{Y}Lm6JZX;RrNK9? zm%2opl*3`KM|YhFc$6n|R5k^Mu{WrB)QnUOe$x+_+d ztOLZ&PAXETK|D-ViTDv(n$Ctu1xiS7Xbco-@C9W=yZ}(6Okz}|fK<5^a5FFI$>{oq zHEXV;9VGDscxf8(1#uAJbc#{4l227mXm>{HsAd+aV3GEb2usF~BYBhzUW(wQDx?&# zXt}BIE`-fHquA@Ec_OL7<9!d*FkO35$NJ1f~_#DdvB^6)I9&ztAMW zfC`d~3P}Kv1q$LoD_4*(%7j$$H?|Za;sXj5+@L5f0YVi?FNF%#Mu+l>$W@^(*cB^~ z2}DIHio`@$#myA$-O9GX4{J1Ys{pkduT7DukYHv10F9x|U*}3SqNNXs^sfbW2Q}LA zt3c^|y<6ibX*)ObK0flfMovM58jP!;UCjk`~KK}sEiL6{8 zRqmfmiY26yh0!ucCm5z6qH;mA8PN1hnN*2JsoNWwzjYeEvb5ULHulkauAU{rAvA~G z?TPtbba=_!GlyCs*<@s1^XVM}X*MsqItaB5omK2)?-PW9?yi@DszZY47OA+$scoxl z;qr`DVC5q&)lz%995jqwA|JppMy(B&#i@os&OAxx^=gBZvk4UDNlpq+k|*Iy;d0Bx z3KzBJL@Fc&Pyhf307w97y5n>$mBH0X(y!`vWl-rKID3nAyIY)0UOK)B22?UNDR)BG zo8~nuXEgFwmmb`=`)JsTpnIHy8-Q~K0eGypL09x`KAe>0h)c)3aH$)&DV-p?OR9bX zS_Ue&h9J6AcJWd=PW@G3wX{bV%B0+`N%g=PByQ*nB3)xPbz3uxamQ^aQ?qbilC982 zP&M@~kQHqcO4jMxT_tE#;z+%?g!3amO9DkM4gUZEQEEP!6-8xD`z-{``y}c*O=7-{ zai`pmdkkc)?Lp>7Opn;HqdP~U2GgYdk@D59&7me4(481F5Wr z#iFdLHACYS^wY8vI(O$`7r`81FhPrQeho6+7^HPnp^`$ORfT_uFqf*h%^ZkX-B5I6r41z`OhEW< zU9~`@+Bn>I7m==yLn3P>X7sxh-bz)p@(oB{;F(<{RK7__4U>r*nN&#V-tuTUNn22O zq5DE+bU6v2qv9DS62~tB>Rd!Teh~B+W)_mmSgBK!F*>eF8yk{N$n>g>RHQSO>cm=s z2tluF%I0BNV%fc%;V)D>O7o{C(r*smomax-Br`x{lwBJYLcymA7kN`0=WyxTx=Xxp zeIqJka*Qzr_y7V>r7M^i=T%d9W*++k!_r<7IP$oRRhfvWT`|aU+M&s%4?>uS;2Z!NK;^$h2O=V~8gW-5 z9h{fFp%f)yy_|{i_6h9_IA$$t{{UAbXIVbzi$}5-1tbKnRpW2>Shs!E1AS0dP~+lQ z6s{zdrLOTeahdLK-AGThPD?(nTzNaSPqI@An;C2Q_@|p}Tu-CR=`GOH=lr3d?Pqh% zrP%AEZ;~Gsh5rDxPr)Fg8ivkAxz$Z$lkh~mVdk8^ zg-=J`XVl$;+Fe^jr(DzUCVbTHvviOz-3BK*Uj%cb&m&Wrn^LD&Sw7|}8XBFZz$`<>~j4f`pX7#T6`-J_488W@&{>L zl{D=|Z!jW+RucM%M<-+N}3l~70||9nR0s>atOHzRcikLv`kWa zKEVR?2k(Abrv^s zUgqDsj!Q_b5d0Z#<_(qq01>L_l~c_=HYtp|(exp&=gieMr@AK5?PWtYmz2#TGJBL` zPUPH$@p#`TVwv+h5_m)zKS|lIiK{Z?H@B2plsjp{r!oT*C2Lc-lnfP9zzMv|h^feB zE?l_-h-1Z77Zv8h<2u7aC2r`(xK_t7RB@p+$Rq61(Pi=!@w7DavY&!vqm?!7f`~*b zeB7&fJjr8sHrtG+xLK6pJSE$m57ONkqg2**DFwWrv2p(ZCF@ykTk{MeV$tI%Av(LPy6j6!&L$xbrhshgqV56b#{h-9>j06@je>;Y#bCrhc^8uq~m92GH1EJ4|< zU5S&oicN-0sX^E+2!elmJ4wUkHkDjMpwl!h?C6hFwaZ_f4b{2$)|YE<)0<&NQ520V z$o~Lr$~LB&l*(Mo;ZXz5F+)HVY{IZxh2bsnzLONmHIhRd*-w&vKs`z7C|=ZPNU zBVv;AMdzaGQyBXyq;RUCTFGK8^hRY_P#N^`MbX5E4 zoG(LTjRk#I4`5BiBE?@-z$jU56PX}Ttz8v46O8C({{T*%*^+IA2N8+MpkruPJO!fo zg5b)bZa@_8%e^ZX%-S>OT)S`w2NQhOcsk@-~y$pjZWYdDp#>PfK<-* zp)Ks8R2^Xn^5iMKyn+@XV=Lh&K(FUqZi%ah0mtO+jl_quS1W zUhL@C5JbEd$rFH)MXgF7Tj=%~2Mv~Nvs|3Hn3n`u#zi~)T5@p_HK@L;TZoiJO{zpD zDZ-+W34);Pj}w!(iJsxK0nY|XU*a-pH$274h9!medQYo61IS}o+LEn zgyB;pe8Tfk6)i-1^KlZ4xNY$+wTa!i5aG0rBNLKtPBFZ}V$3J4N_>D`V6!lj*QeB+ zFECt2Sv%9}dbecd;fh9Wl~ml6%Xlva0&1zqp?n~VQ>w#xVm%>rwRJ}DuSDNnW zq@r^B7Y6&0Enf`Bmw8LYe_oqIM`G}c&1ZDl4{|TilP2bc5cXxQy33gqpUw=ZH z%Gah!{{Z&jsXEphWih0`GrDS%CxXeU?tK9luYpqZrhVR|6+F?PH!-p?bN$fr4~y?g zUBX_NiYW_@Vvqj-hj6Pc$Ba#fma4R=FY3)&r}|K%j23?tfBaPWo{7~PkeF~NO8!rg zoyeev=!R5YUxwmtk$bqxY+czlhZ8~MrW?&!+(l4X z0=T%NwN*&yVPRe=OhH^a*CT=cmv#7TdU1QsacgXrAn3)5<%v7U3JNuB@Wac2wP`4%CZI7;E;oC~3Df#+w!F>Z5SdraK%-xFOiOLS--(vb< z>A!icmW))s{?3`3&p=?4FHYjL=`6}*<`VseB$d0`>hgt=>a&V~{t`?TU(gu! zrmrP7o6>wO-0Qhxc|=xh5i#G=Q%uTEaye4^Wf8|NXGkQTxi@mkWR6Y}CMkoCZMPAX ziOI-^nxEh@t4vaAQGG{}hoWYYT)DC6Ly9o;OnC6!%<~UKTMfs7l1Gp$q7^z69^@&^ z*OGA*HQfa}knuBfkWp!@=x`H~o+cs5!_g6GuIo{9QbCQV9OfbrFj{LmD|t}!Ki?3I zsf@%Wi3fythT~d|g`n8PwVTdeU$o{`f~gyf$d8i7Q`W@&>Z)_?lFWyRFSxTf9=TZPvvk+U}NFwu9W+5l8@E4V=<{F-} z^D3ih>FIM^i-eGTZ+kZ~{(xR;#KoX9{!09eft`)o?1_t*S#?mah1zAp^Pdx>lCb%$ zOo6`T%X?KatZO=anR!lEK@UvB3(XgFs$FtKU!s*?}%y zW>l-i{X3z>e}PisxTC#DB>Uamxfl0kq1RJJ_T-DZYo!%`s9WTA|lAdtOodtE|z-Y$^B^ zY8b+uB^gOI$4Q!};3{4|5rrt;^W?L5T<1h)?^k*?II^vurRfT;LYJ>>0-r(jYNtt3 zFp_@I-Z6W*iF8=siS?{H>DgqOmx)@IW6oGCiH)7v)lnKlbP98CeYf@m=(O}ELkYrOeeo}w*9(FDf5+8BsAn{|9ExIjTX2Ft*SnB&x1Zv)z**3?T) z1QpHF?WAEmGLWSsf|G12Vuo=R>K;joZef-{%+J8UCD5s7afR9hT08Wh=bjLzV#-Cs zb6()2tA_VdR8&EoVIT1U1w4g012Tz`oQIB(Dfq6?D!!Oh7%cYY%MGXzGB0f2Ap>)) zQq7hB0CW`vWjJc*b7E>#Gz{9alzSvGWgsRV>7m6+&>zVMCDEYMi@P zO~cqFu1YYdPOCaz&=n-3H+MeQXCO@%2^)}8@P@mXs%*9NHZfI2({$XmSgN9_2xOH= z*N9>qbK*f%HyWsTn)+H~Ig8?rC**83$V=NPB5)j=QpULndvgB(j&hsi4R^qD zm$m3joV%0o(J6+%Rfz85+nAl@E9&x$hW`MpH9TZz$GRo$o*1#Xd`EWQf(rs!WW{zh2_#-EIboQ4HvsAc68pzOqjG{*(}vW2|X9EauetMU>S zWDE5RLBr4yCXJ0ctGaNh+&_?CNyFt&!&OrWsskbw000O@a+xdWOLzJOoVk;FyOD(2 z6|zw_bB4lE$=^*l`<5a+nC_Q3{1Odl(V6V?1 zSG%rHeJ)Q#gUDh%BH^mFr*>BkRZ6M0q2Z!w_|#$G<4})c+?P^80^K1;jG3BPt6H6g zOQU%e-q_WKrpjbQ003wW)LW%bVr@~T>FCn0QA{&ChO3|4nUeL~Qx}dsCJQBYYKJ;% zK2$z*%S+1C5|1aQ->&*W>ZeuHl}&icoO>7$QW?udxoV{#y{lttV-?pe40_Nk-npY9 z;cYrf9LA>V1ddj&)6y>t+Xf!9qZsogVp`=q&M$|$GF1#sX_&MMQ<>p(i+Hz3)LoB2 zYim=;L&M^#@+HYOiMTDFbp{%4={yFlDqgruW$q?#EWy@XBA)7;*|bbIpkpv5R*0sg zSvb%$R4YeWi*kA4S+8AOWNMpaalV#Zn_;S(LciA&}$?^`2YRL0<{aq~=n6yiNv z)6-$nTr{eYoRZ5^%v?+OHZ<1~{Hcm30K=U3Bb4Qggx<0+lK_^+?A^*T|+^gIQqIaN$Hs zDZZ?$b)9WSv@J7Mq)M`yF>sY-A*N!b$s&dr@SKO1#X30#VQ+=#ujhv&Ig+r85#0#c zN`}Fz6lwYEr&W4+?umc!2EAHy8IIz397KaV4P4(Jfz1`=m{#^_E_h110eI&;OsW#{ zYIsF4FBIctvC%za-QoC*oa;D>wfmTu*g8y0o9JXj3gN4m?DHPUa{C!Y3fjJ;qu|6( zNwg>@932S76v@IlaEdv2^mKy;hW;H5&58Z4M*XpR24Z!+KSaQyLr6$yrF%q|&84J1 z1B%Qw9;&C+aOg_bBB|9In(P9ziL^%&wu49DM6zVz3$IE;6?IRdJe1`6QiydJ+Nsy= zS+-!+Hf#}12I&^!Nv2z+DF>R16;$0sq3&;JQrA?I;~ksFZhCn|M<-NI)={%}PmZb# z8?M&&DY!aCRDQ|9+_vU^aK@ugvL~Srb0qs3ZhdlZwKqeV8{ICJa9=>j9hGv|BaD;y zgHnqJa&NFDH2kJ>0dI5)b9)an?q07Ps!~&x4H-S)#?)<1 z>6oH;syk{6eR`D6_bl7fbWG=XPn{-Cf~rn`L}Lv2rH#iWT))DY+xr)_sN-CHX0;+7 zJYxRRflFMsX^Ohr4_ztIXJu!|=HmUlne5#T3 zV`EFlRpi?=F72u%7^k6bFC`Br^8Ww|O_y`hjsF0I?S{@7f93)&cd9AINQzwSUGbCA zgn8w%TPkCCle?i{O0jscv9BHzZs`S_in{E=da#J)rZ+Xk-81V~sHKa>PCA_`d3Kdt zVMQ@mxQqEVfX??Rv&w6%Z5{srAf{ECAE~4r%fQMCsD^nr zHAAJhJ8oD#VAmz1+JfDuqZtz)vbnI#?4|TdG}bl8jK}&+kd4_(n4sgs_M z`xe<69jBw&5Uot2B*QbZ7tu%&ivD8)=*(Bra}dqzofPH#xqCT#J)X$gM%stDVm8_- z>Dew^p3YBXFSMzReSHJZFJtcoYFWd2b3!>4Id18R;Z)5cBYC0dQxnA%tTNs*uj<&E z{{SFVtP1M>t%r!9#=@H;JJ)Fvs;}Nq3ejIZe&YoGVTk>daiT&Tfu3n{1#Q~RbRPWUDGw&qhG0DD9Vp7 z!c@~GO|(K3;~J?)WI%7I<)~8IXqhafElaunhg`n!&ZEu#>2SC%A00hMsWw{2bCjk+ zqa$N8u^gqM>P$oII7s6jBABe6Ak6M1GY!iU37Cy$trj6}Dc5Gz+C|to3@3=DXz8}5 zWOCV+hr7vgD|9D zRW3|sBs8O#c^h)gvp6YF7Xu$8nfNA-m}a>Ru&ACYkj!!$*Z3?FRtapd#x{Lr^(k&~ zyH=@-v_gsE8;r%FxWr_jrXpNs(va^Msm-pD(?6awxY$Y7IHSYM__Bq@X&xGqTp3ji zZbDnX&=o;cCh>-3rhI@@iqPl{%v+fxZHiZX6$)xJq?jw+*5H}h%lIUPPm5E=bf2V& zhcQ_RYY}!c?bJJ_BO6AV#w>WnX#mMA&EvM3B~*p{0)LVk?rOU;jnTf2I%)lxjl@z1*t+}#On)Ir4KMBs7?Z?OV~d*{{XOx zp^p(fNvn)ZSO+*#?NcXWbejO6D~{2L7L|g#czQ^ z;xN;ip?Gaz*3|p%J1wct{>fMBh~l9`@o5(Ruep&l+@HDu(s;$Qi2|4Milm38VLU@! zamg01J(Q{@6!nxjbP zF2E|#&A8MDdaYIY47;AF_Ih(WqHO&#`_aPgPv#^ry?e?Xi%tHB3-uXnqQILQyY7`h z-_@yyih6egh!Yc^cPJZ^WTW>gMd?4a%BEkzkEhCq=~W7;ms!Of`UrM6AhtKRY0rFt z53UtbYl%O(5Q*yT%CUM3EQ+j`@T95&A^IeQ=3uhelIOoq&0 z`WFOdJ3kS3IQt%k-3dxF$#N%i#h;*GexTvWy!AvY4Eri8R;v+CncRIdCur^@D!0~U zLqbGH_XHfrLJ`eav z{{T!c>CSeQO1PxrPZH0I~!O{HaZJ8z1s zNN-K3$qJJP#8bTVDfc(4?zF|DA>z!lSJP@NLe#@CCijfjb4mXI?=-(DfrM2{x~3}L zYRJ~9QWzJs9%N_L;KCPss>)p$0? zG>3w%CG{N(5V02&_D7+t(#JTq@Nbn$FZ)r3WW$-noBse2H`u<(PrdYAGH}Xr262$MtKKX2>)%xA7^z*tF@(C${zXBS)cq-K^G9ruU3FQ2i)XRH<={jIBaS z=Qwt05M+?tO0yJxd`c5Jv^~m<{{Vw5E~1j~p41AV=w1sPLdxQf{{WX@g}T^$X^q;A z%uyffvu+&vVt8swVOXQRc>5A5x^^)G8?>9BEWr*=JT^JR*jU^OGr!!7zd~Z}!?aeZ z;KEXrJV*5y5)?RQY0=mY=BxE6m)(iF4MN;XwQuen(Ogv!gQYT$6P%$5y{;j}3k0rPS*9(#!o}JTy8m@JMV*Br?MC zeqvD9&zjI8yAvaLk zY}&K9u5Udm*6qh8F@GzhElABe48_%)%d>j7f=8{sQALRvr&?bhgr2c-Z#@+2_|4OJ zS|aMPPQx*YSov6mb@CjgxN?{l8x-4Q7!mWe2ngVzE|A0Bk|KQy_B9OjUYu;)gCB@o z$$A06HQj?szmjZT8QQ(5FfnrrMX6K&0BbHVS!8xfhNJpFhd-NX^)E%;4{QV}NNU-2 z%3hM)y8f9&a1l8gZ|IH`{O(lxk{aFXHyU%0^sK-C05GQ+o6tz2uCCv0J>D776ZIWm!tEI^&U(1L9(|+Dc{&@98NxMJn;`q#uad0 zhAzT!Hh;~rDXJR>es8xd>ZpqU0Oj=pOYGvHBZTxd-7AW-{%whFZgp{Zgnwv^d?O&q ze?Y|!BA~k#g_@z~y$cX@Tj*O9lOF}`G_U+6%poE>yq?y?VdSj_vq}x&u0h+SG;jW3 z$@_0j0}9q6l=jTaGgw5Xu@#lQncAhCugJSl6)W9oN+*avs$xUQ&`%%ocC@!eGSRWI zQc3f%^UN3Y7Z@Z;mHz-#OD{IB(4&yt>uhX`f9!QMseSB*Kj^Oq5Rd99Mg1l+^2_ZZ zh2y=T>NM_f@nY8~hD0)ljJ3{JDYdsA)eq>4Xm1H3*~$%SB##`aOk1~A;Ur*)>Y?)A z^%)^zatj;144Zb%nmmrLn%6OLs+pYYQfEd@BO2?3w0zck)4>-W%%}|L-e;pi9=ESDwXgq z#oWJbpT}ubBl%`~F3BsqYJ$Yb)PG9llap6)ZO*c-e}yeN&Ew6-D)kyhId%$Rg-vhj zn6WZQB;R354mbX=?u4U~{{ZE!-9GdR6;eB5YA34*^DUoP3tcz%VWlOj&$<KX~BTFopPga!q5+Cn{?J884`H}_gD_c91 z3}}Wb_NOM2g%GW7a`H1PBegn$$E#LVbP=!NcPCAZZ5J)g!brlmF%#FNF^M^ zUr6k2&^A>~?wYeEHoazdSDQhuDrGMT%3Nl~{HX;>e1ibQ{{W>X)L4v3k$2l|wt!)gIYFXoBk*uKLYc>zph@MG;S*(+9 zx)43hG>ywG{wc+X`W37ynZymU9lM0->uYV=^Ke?!NM17q{9_*WZf^x$0W>S*Wwou2 z!&eb0v#2S!%AOS6)dO5AswG>esAei! zhrxR+say3)T_D(Ov^4H0Sn(IU%Ajo0&uMcM^z{=E#N6U-OMhrsjE0@WRqNhlf*8XL z)C}CJ-b05_7u1>>{?2MtXB<`4gQtlWHCdCeWV)uhOeUl%N7C`}LX{R&w!YmJM7o$# zn4sH8R}(}%(>8=IDoJ?Dp6WyxgC=VYSX6CFdbUfo4b|aB#4Dy!ABtu8eO1r~K^GAl z2+cUywrMkJj{zi!%M(%xz^DQV01Q^RwORts(<$n@k(v&V?hHzNcT1m=LjJLTFlL;)xHsLAA~t+8;rGaxm1RfDlX&& zWkuB@9;g5S1vsPkL#ZQ?GY0q;yX?x z36Dj4&Z0^-BQri}$XQh-NmY}PjawD*W8_>jZn=xZNmCSiG{vL66Ei{{9|^0-xD|s0 zi*T$EFUS}y%xCwT3n>XZjAo~^m`Z^&Wg}vsXcoK<7)J^ zs#Ax^+nSY9kH|unkfytnR{sEXE?ybOCl8QZLDx9hoSI2gq7iD)tkX@hN$8M$CR$aJ zNR&LRX{C!*sp*u58>JV~&Z=JJJ5i`(K)n>y&{{=ttC2;(=z zn5^8Jg4tbwy$aSl6Pu8%ZB0}UKIDdazlwp*G2@QA2|rR7X7*cX?n`~O3s4`$i&RG( z!DnFis(pWgxxI{9(~oeoYM99punTclfn~Kf%7}0m6}uag<@=cw z7UzA@?4Bnj3ukl8T{kuKmLbZ5r8ud~Lryj)hB=E0Wouu8ol9ryTfIRNS~GM7jqjQn zQr*b)MOx9F~o-`)0JNBFZdsft89? z{Gkl`LF>$QRZHTy1Re{fQ~VY#XS?W>gzC8#VNalT54n@XxKp)>DR_w= zjBf~jdBt%zJ<5HqUKOmNzcS_QWE@vLVy0%yCZ$M<9NLxvWHus$;H@?M2;b%UL-tM< zzKQTGQOb`lzQ^MoR9wz)5$N7T{gW3jv6rwSh^9?QQMu$_X^K9ttloCIl&9q7U2{zk zI=fdyWQK&))TEYi9ER|xw<%W;Ez_uNI=zUX81s)G-gWXwK}w?c9%?>21}~!_4)yvO zqg~K8HDkATr&8Azfhv~)XR^w*XR-*@@j33nF$`bQkNvdp{LI0s-P~5uP&zT;9Fi`JHVAy=BQH?Q~ zYQmSN%w@iqpRtR(T3ksVl8M|GnLKwbBj@kXToZC65l!g_{$?JP?xip@rlTr zNDOFX0)nbaGBKbMD4+`{01|!>0001yd<98zWc*+p`-y-xwNg0QMAc8;~fXtLt0AzqyX#gz;rYXh?2~5$-dKFR}IAw^SV5((szAIH1AQELj0FXhL z0w{nA_$Z(V;3t7ORzD9IsN*{l_fDnR9Sg)RMS%%XoMh8 z*Gvf$Hmaik0FY~Hp=9pwr*r)jxk;;4bsm2)QJ486vZhniiVIloZ5yg<~6IdiLj{*CyPR;vvbnPSWU>?bROYq zUR4&gWQCzfc7q}D)lJMHKBK3)w6<&F-?^BfW+5YCl^5VYT2O#6Y zZA>re9Z_p}Q2I0(yiMMUz@mSY29Ml|%O%6qHj^M?^Gw;GUpVFwVy}637ZOyZ5}%(E zV#{twWpwFM_K75JOPLm^7tA1V)P!avzo`)OJ*4h&Z}%4Gmn!&9YNdZrg%?_5)MwrQ z06xS?ZT3+em$M1Eu9zlYf4_L}zInom6APVHs>ffUAM%w&Ghbt2_FRrn{zZ%FsxaXx ze`PDZXDKPSSK0$oo3u3b7YF|U%jw3Y^vGc0xW#ZLFEwAOOl{XT{dmVlxWM>`=BRoJ z4ESTPq`VjErKR@iOL&qoj_TB^GW<)^rRWC?z8(=G&h6B)I<@;#d3zqvs#R}ruSzfV zhJ9JSCc|*(P5mZ3@^)fHOlQ)LS2rm|PGLP=rLB{O-sC8;R^^}jM9~YlP4F0sbD#T# z5S|-+08q+^NCJoeQ2-JE$p8ve$!zwBo3v`eE{}vnr*|T$5HBR9;~3rvN-Bu%<3|<5 zAz?Cai5Ynv#kw}NgWT6*5tc1tx(u*BIL-Z zETK5a?1`be$>b`VzifNG*hLM-PqeB;=$`Gvl)`9kGBX|CdU3CX4aOujPG_jHfv?P- z;1{)3Ns2)TT#4{K3PEZXC43J}Kjj2W$w(>SdK7}hX<|W71JdpBZ43A<@Fpn;7LKl- z2cb$X;hA-I_#x?ZpNu#|s)&EODJ=c5qd^p#XN}a&ny(PI;%TsKY6Y0mhZ=T%!5l1O2702dSSY%F{ z;^?TJf+@(gwB!2HL1*m!B@_Xj;8MF0vQ z08Rj$0Vf;)1PWhhc^=f(aB{s0MF$%Slwla;V{E$(dp-(xX7A>B_BkByn4L>t$u2dU zlAjjn6Vetc_-cIl8;9rwl*2&{pcc zF$PVFbs*!a(&W~tW`DYoGG1`mY-6(rq%_F5H-634+$SY4vyO^%nTkS+dj)E4vGAfL zNq8W+nAb917{OXXs*;=987YFpB^4=Bq7n>ffl(+*6hftId36ul-Kxo!iDLD zNw>K^hncVO##A#&HYX;h(3jcd0cuOJJBdM1#bLEOk3KN1Ockg=U|_8zHy}ZR)Re>+ zkmMu-S;%=CXkb~~OQIyLNlontDqduuRHTCdq@Y%+q=0KHXh;Nd0}!O3R-g!UCZmn* zxj%FYwnJ3*hIw;uT4_>eUh`LqxkfHZpEK#dMiUWY#DWP@=)M8QyK+;`bTg!?j_MP6DrA*7^5EHOpIKOo5fO~!NF=oY3RKN3D#%s0 zdt-*PgREpDE*<=$rReDoopjR_>hRSN?cp+4Jrw#AUJ* z?=49`RyW0HuXS!=^z1S!MuLcZ(-MTRqbiwnlLe!mEX99xNXIDIdrTp^r8tIUSl|Sd zF9_Mgbs4-ws<%Fg(lIX(zziN@I;VuZ=8{<{RCR2Ew^c92FZ!y0F1rRWqaN8ZE(|*&SfLFL>Cd zi&UA+N~Z9TJ=`zN91-+2oZ6WZ7VTCrOs(xs$Cq}Aa%sv_cV8jnnFHNig5-*dV6c@6 z_BOG(uBUKT$XAF$K39N*jw=)FxG6%2q*cu+>&p{=II}xKvf7Q2sz}tXH@<(hww`-@PG-RP2oQXPSe;1BJaTP9sgjh}VGNVxyAqr!v zz@;xJ%9LiXXc9F`OQxr4whl5%oAO7zD_pWIMX8C^7LjMPK}@w zgH|-m33RF{b0wn2S}GX2j<-PCUY$cW&dy;^{*Q+4@+K#Sx%3kGBh1cpm9Ujjr)FCO zM`q&vMcGviL72}V;qhdZ`=Uwl^OuYG%AWDZlbpsM8WS@@g1Udk zij-*Rn|Dq<76lpC9J$uzQ|B=DuX5W{oe6u6vD{x^j%1{hwc53v)-6AQ5I5n~=X_F+ zJ&8{vn5c?w%1bIORPDzLqRqOG6+Vhd9aDUP(@g=($YPfHEigOg#FAqT{5)9Nts-9wZj*!(Lr4-Mpn_934*2%_1 zZfina%XF`{5 z-RU`8E@JJD+>HDQ#?=(##pv|xx33+3gmUhXhi~o{17-#BqD*g1QEJRWfbL{Nz~*BA z0L;Ha2i-_y3y7VPS$ZWW^LZg#ryy6El5}n|^HpEZ@_IJib5#(zJ5xzeO*|&hEyi%? zAg}5V=+sX7o@d>6Y{&baN*Ny(5trPrKTXlx$6aN7573WleMj4Xa4oAUvEv~mn+mp^6r%rxt(b5x{dvIwK z?%hjz_U~+K`>m){=Kla|FZF~`OOpqbX+mpUwH0_)gHap(rnpf|(+!hP0^aJqtzZ4w zSIe*CJyhP2g{7TB8vc|mxn?AnymcQrH_HkU+3=G)P`gt$fm%_Ila!{aH}R|Uh^}Ia zt*J!L5r^VUwlfhvM*W7}V@ubOsnc7as5UhUS8ftbqD4vSLltVCUT327QVV_<>J;@} z=BqudDfiXIg<526g9!HLdg?k+Tn7!3CXt4$d6k!HxlN2W8JvL~AWpEm{^nhmt2tj( z7R4^xxW*-!G4a(O3hCZ3vkI+hpQ5@sBmI<$!|5RTGm)#Bp3xgXDq0h144FFu_;A(j zF?)!L7{zO|C8~TZvwB*fs@gY-rr}>nJ3ap5*}2s}3&9_LaMRo<=`Pp~CbbwN*=|MfV|DK2FO<()O1$ml;gC zs|-;JxtfZ&wkn_8{?ae!`7G5A>}yALAKFS1HYSIP;x~qDQmfX1w=L0#EzUEqU!kRD z=jSR%YDmvaJCj{r63$gJikrqUhIBiyWu>K3keO+3$(OP11r3!%i_K~PnmxV5JEtHC zx}4C6govhU>XIB-kqVWyBpKAi1wa>2#03E;gCGC^00{s90000>moH=qYf6?8?-xfnGflGaaIlHDBzAqV3 zt#wbHV%Xe#b^S*V-K<rb_8iOJ?&poBh=nxS+X5aY9&O;=EzBdIS7M^J(@qUoHu62xGhvKQA zEnjqXrrPr6B||J^BQN(T#+h)?CyZV&aL-$X^jcF6YC5++#Z7EgPI8pFK@{xuM!DCs zk>;XEpNK3)Hde^Zv6(62bS{jV@~*Db=BJJk{{Rr!Ymr5bPVgp%Ha$DZO?)UWdUuWp z&ZHu$5DJzgMM80ng`r5QM(REDX?? zSg2)tRB#1Kgi)X{{R^;$`FS9EKRN}igIwud&1N@kfzw=Jqj&EE=Q$LL1Gr8 ziNU6AeNR#imbBS!XQ*6;btkwcB23VqGenvVkDAfCF2<95f(N^tnB-6 zXikd}PTjGaG^e*c&~GdH@Z?Hq!WW$zgl5Tc5HEyb$BPkSI*5K+={7Sr(_B$yAS&Bl zLyee2orn97>MNUPgLY0Sgr<+|jf+|^5Qidl!*!Hz33qQB4O+ zIlaU{rD4zjAYW(zz6NCA02u%P0XYBw000012}9Ab?m0~SgOFhu(#2U=E@!~{J-Z1m z+wN|B{M zsue2KkZY7H9E67UfUQ-;rqK~Xw1*8dQ^+V(?i61|PTkxuTat2#n6sznDUA-UaNRR; znmF~wv;7sR`Jeqq!rg=7t;7?=<~v%V*0bwGoTQFU=F!pfTij;n(o4)R=*vaO{R7ZU zT2mIo85$zyh+)Fx+ExO9ptcHb zL<}J@(K93hBEm=dE;~PQwY@W)j>};-f^^*`sz+}QXF5e)E;BQJsG(Aa!}@d4$4N|f zl%i6c%`du&z#dq|U!6swb!mx%obk)^9JCGL6iPVtT@UFiXW%1VzLFY1Tzjs0lBxb6 zLAUoESIF7s=~`*y#d4&^btPdO8~GLBkCguas1j=EYN&dQ0x~)1ihU=yie0p&8tPz; zjAv1W#XZvAF_I?_WPW~!{g|qU(c0=YD00sv3C?+i6fa!*u45CX!kzm%tvzBXX(a zV7odRHpzJ1{0styt6EqADG$Ix7P+wh?$jY zYrpOWYZj!W8W#6fEe|vQ09fm4+D*N6P#fG6HX5Kn(c%upA-FrVI0Q{dfTG16ifbut zakl_Li<96^uoibGIJCGFmzK7)+y<|?}f_AAxjdC+=>%A-X+F{22>Eq&!I{a^03)>KXURCYu?#yoUh&$pj z{H$}xU$S;KPV%2>H#0w$`kl_X+AK`AEK~sAKj<{9fw?lT&~ z>kN8s3X<$O=-M&j;8uTPo>^6tLMz4EqOt{=MR;EatTF|F-vs*bK6-eU;_vrEwn#^3N>V^vEV^z~Mq zvNXgexH}1xj{hd0%1*XRgXF0NG{i*@1UTj|2^>Gnf`+kR2uoBDr08VlArrD_uWF_0b7vkeQ7lwi~(*0r|_^$v(%sa zuHd&x**4<04I%HcJq|Sb?ImFjP>etm}5*01?-{GCk}vQ3I|A|4<=59iUb51^^4&SI^K&Pj>#s_ag9D3Fc+^wKHCyC{d888IgbwSUk3T=h%H%VxqPpF$4W zzD+FZ(fLoI7%P78NUEAIGl(*-SQ7e|^SufUmwep$HFm6V>=1K)32#&!9>dRc0xEBr zXW9foeJUdwT4+UHU8-Vr)W%A=*msa@nwx=|9E@x0sLo!`Z%F5ZNAX^Wp3*28wc*9K zj9(oaHM@ov$0^H!8^4>~4{03Gz9zxdJjZeoL)F7ff-SSB4cMT5v`^~|gsE36IurtH z*<_ilKZ%b@wB=EpM~t_7k;-!w-br6nXvp$=1Xz-bQX0rrn-<*8#pS|DPb`%P5^XQ> zzKe{=M8*1K9h1ynx7OUm8>hfwttw_ zScT+n{=i)YU4jSM!$RV>q?|(Z3F`G7i3tSeImKZ(J*tI1+>ntwycw}%Jf_hZRn2_c zqB1KkbrktY3!rKcfs;l?%?XeI-~>hBkCH{s-gLej7FSl`knoNoAo(PqNk~q|9*>pF zTg*C|FxFG}l4#7}=NAADx4}Lc0@pWJEP;H~aacg*yRH!>#;?mYJTJ*u`OlCicf3Ql z9TW7Z%1uxI%Yco5OKS<+nqMt}V_p(cY`HD6iBfW6F$JE-dE1aOrgN9gFb%VqDVQ|S zz3rFACN<3H-uq+KCfB)gB5tr)cR`gan47(k$2H1^dwmI~L4}?YVZD!1De=9qro=uC z#c|~z&6QsmeOgQnc!~WI07wM@B8X>W=y0_ns;~h7{uYTGRKB1T!wE}H8FjT*M?|ez zdF+nQ+ijXz_LRQ&wj=;C==L?&xAz~=7r0jd>+F&iRku~4gD!9zHKiE^>#1%1v5tIq z#=DH^SmeQ#XQuY{)2h3kt*2h|U(PKy%2WzJL0dQ7W4;Mx4ys$zMr4n7+q~ada-uud zsTwp^HPSKr_nk5P-R?zLN)=nKAj0>k3(B6YE2T3mCAJP&%nY#|GDsH{(% zqPvXtvvf+z3oIdTQNLk4&X>t*#2(}bIDYd4mhu~O2=qowjSyaKK*{AdOk*LV^36g< z?ZbL1@})(Tn<49XgjOve0zlsN7oc?sAfu0XBFrxP_Ver42`WeKnqTUJJK9B82J#$Q z92?KmUO4Z1Dr6X4TNKd^Sn*+;gZ^dNdGwXRM|*SkyaYF;W-&-JQ#hY(#bz1>1d6;A z&fZ9+mV_Hk`aQ?;TdM=oi%-W;U*g?7Ly1l>^7yD=0|5H(dI70eiqmBD5ddOCEL+O= zPXUcjdHw+$(#WW}vCYXx^>Zt^7E%v{znL5u7bo^+Ej;Nh_u`&a*cq^C<19wpz&o5uN!P*vE^r?=9XrM0%W2Fg`|ug?r?8Bk-D^ z;M|VWPH#zFYlJ? zO%8!5HM%~o8Z6nj)GGVj(H*an9flJF8MIqSJ78nj` zgc{DuB*|~87R_~)R6#^QbtCpf#LTvVp=XG(%F#>d#NCJ_@TKexorO$XIt^{6k=@pm zzI}9aImY0H=v-L{hDaV|3?o`Gfn8~Ieu2suDKh(UqG;MljJ?jQa(YE^K^WEeUoNpk zYAVW9P+;RD011GaO<%J1Gp-xr)$f{v$_#468JXUMJa#x4{26A(Anch8AmJxJQJ`$$)un~|Q( z2E=1n=%3NtZuw}?PxNl2u>DKfCBS1JR}5y#luntST}gI$$#AA)5zM%bJ)DZo0uq6g zW4DSJ4XVJAY(2$ha3l|%a(t@M!^?65$roQdjE~s{&>FySz?~V4;)6pQ)y0=lYNxKCr#C5tq``T(={QB-Zo5OPc{6 zLH%CC9Y{IH0@@S0XDaBu=UNt&*3JG*eqI)DY68XBl*Zn~$A|o(>N|xLY0G1-{BD~c z^J-m8I=qOhPXfmK(3+xE$1GN7bh6?;!J(`Mljx~^MODitv;CK0UEa1nP{A61U-EaU z74f|8KG{~}gs-G|)H+naLMkc9{yEp1y$8x@&!?edOhZZtV%6k9e9q2F;u04j*ODUb zc1d#u|ISyTvLsi_Xo&MSM`1W&3v%jI4R?Wn}_6C)3GB2yPX)5=r9KC#%((x|w zgxx-MG~qVt7}m64;paskbqEe=;qltE?zzs@?mFe7rI_r~HN3AtKXFd2x^b(lNN6te zb-p+mnzqO60FZCIn4>C?7sQ5DL;DG9d1jSgkJ7paRXdu)Ts8TQgL6s-@!UlLTG;;^ zjBQiiU{Bm^m{Vv>&XL*N_;hW!~YdhKIpd&w49Kqq6b8OSjGNw8`;JI;7b0=j?*mHk?T(mJ-1Ke1T#J{oV|*!+bGqm z2l)lc_n{4WxAVSguBNI(4$1LVAn%D~STn6w*pe&5Kr*V*2L~1P;F1TxiqzFh2cNsI zJOW(WUzj0PTK3G6u7tof7(WZU)Rpkm{KY&l(QZJ8y5JR5{dN|~d^EeovMJv zlH2b$XmRGB^aew=`*?mJtA?@a=q!l8Su*_l31ldDD7O}M@JGjpnLi0)uY%gq{33=^ z9wjh1&?CW{!koImk?ejj;-RrsD3J7=)M~k1amcdHo!dER_yHJG<2OmKff@I~%y9^5 zg1|4(zxSgssLp+CjwfWY48~2B7X^c-J!CGC}r~EHG4m z#*NF;E54?Gk6qSu^-A?VE5k4?SdNTwA_O>d%lBKGQP??rr zWx(}7$YJj(l1hcmP<3?=i&HH^G{95DdvDu^H8+9oH|P8c)wG5?x1%{8V_}PGi8Rhi z{8EzhP(2DSaM$i!;cuE|e~3fE70e8yR1`7#W{xJt$2{OT*K?dog4o=4Vv$xD(fS7| zdB2d7r=0PsyyHh|4{@?^E0G~m(5ggI6`Wt*kwrn0{`1LGE3FG1P^m1OE>7PkrxJLc z^U?n(Hj*Xr#{HP-HhRiiJKyHA&U=V&twOD&;JuwWa<0macU^8Pggn(#NF-rEw$ZRd zk7zeap1qN(k?L14Xz#SIZ6ffZXyG|hq|@+TgNI@jmF?9nw|NlhY>BEE=-eJ~L5t4q%Ki ze$|njD(*-}x%81YdgL^ny5zZ{J3(R2$d0<1ifY35@gG$!A#oV6@i^bzO|obQmoR7f zPNusQg^<(Sn+GYp=LD>vw%BO-QzD`Ew2t;`&WY2w9cW`qOzwLG0E?AO|4BO5lW*Fr z6cNO!Sjn&PUK0QSxd19^PO@6|eV$%^V?=3O{l{Fh_)wZ}%RYQG?C;~;O!)KN`2`!y zwOf}RxVZ4bW-B^N@0x5_vkx`+E^wx8-*RUUz|E>*m#SrqucQvnCiODlt>}3eMw=j5 zrYVCg>cvoaQz=974!5_iNm=#4z+alcdJYq_gVO}f`dDX>@fiFinyWV_Rl-McYh8BO?NrgN^QD$Jz4=Yr8cUM06R&B*G!jKwOwLvovJwGVOZn{ z2LK@e8SjW)9=nLcKcp>$yYp#`qx_S2hX9q)#M z1OfmO5LZ#-j0P&N79knc~L2Gq4Dxh za=W~sFsB*`xQClB>ToNkU`&{rttP?BX^><1osMYo^CR_@U)wKaxgb*tg?VoRy^2u& z7_dG?e390GLv8U7q_*Ss<-8RIn4BokPS~zVP`$XVPOE6m1t7YgN=7jApTINm7XThF zpp{(v%Mc(#0?JLM{^e;8Wp^Hl>fh56zDPMAhCujL17!vwb&G}g;kXaj)8Q=u#c)1o z)1F%lA2C&Lo}fb~iNhK?(cXBG0E0iHQ?@UV`0WCv@meVx;45}mtx9wcM%zYK{E3>>pt`8w#|(GJR~_M(&Jj_^j9IHFp0LMVjh)Xl)IAQ`&YS= z-ZlNIC^a(@nDXx!y!u|OA@#2^+dOh)K%=!qO>s{@QUm)#O)4oF|2< z1n1nW=YBLTcRU(CBgzeN?ymaPhWb?rt)y-9>Nrw8hQ+D7X8q!4X{GyYbA}^kdlM@u zw!g-;L%bdo@PLY9J9nJKTaCfZ5;L6-WBr#4sJ!y>xu^R zk#&8;J+eq~{h5Ztn3{EQi%W2z6ZlJ|mA`RR=Z$(*sf5C0>XI)Q1fjJ4&IT%#&^M1S zQ6*fmq*_Yg5;}J_vxHpLk^GeWN(DW7>Rlni>t&!$B#%-axAs8;yKQQYpbC2<8Z7%L znn1&u)j!pCmix3?Zgct)?#!26@iutbCZSYVNz?!`0A&OK@C1j}M{@U|&fxuRC+*J* zZ={ydo2qgh46DTEbj;zal;7d$s9!0$h_{WMF-LJ+)szod+#Bkju=xNK03LuA-fI93 z_J7|G5x!5-t#?f)Kr%_yy(ZV%;o)TWayN|^irc7c4y-f(laWEi=dMzaKqLuiXr9*j zgd}{8oQp{*Q^>5*aLDcE9}DWII#L-jyKX?GnRn<=Z2_YR{cNThkF-SDEaJ7ioC&kV zXCiEi5|zyk+ziI0eAF)szP)AewDpYJm@Q&Z(HC}9{=5YJfLhzVE-TaY^L-B0O@sdW zVOzjK7tIqPH#YkBW7O2uw^pORr&yhcDr|-zx~IXy&7*WcNE$WrLm@j=t+u+Waaqit z9bzcl;0hbApn(M=&^vo?hE~u9Ecyjh24TYyP9ept{pCW1>0XisNHGtKPEoLH97dXp z-HNbSwb6z9XK{#|<5J{9xx5@{u|?%Kx<Q{w_f4nk9wW8n{hOkqg zDoXdCDmO)xi@VI9|is^-F%gR6^zv6S;U?N(;jhu{$z_@3*?u;^%QCzImj z@T42_c$>CaEAkURze?C~JyWfD+53cHU_!sBIw-*`Um--3ZkV$+j1%=>crqTH(wMXW z^(8Fc#hGk;i8}B!ryJ`O1`0#J{C>_0&l~y;l*F7Mjgss)JS7uT%@l5X%B_@DsaAEj z(YkC2KF=MpEc6|5>^L$c?x}(urRhS=r(2#0ymVlAWwpjgW}c{OHt3!0DcK$Znyl^B zpDfNy^-wh?C3K!lIp1mZQ(|10PO(7Us($YwD~hd3N=6r{@bj{44&^;9@{sQJ>#e%4 zapccR9R2;#B4P9}WM|^IbIi8eFD_?->l{Lo6p0S=G+l=mywWXkH1F#nW|_%so0|NQ zf>!~m?G45BXD01VMVPC zn8+^+YBQ}<-w7Q9v-X9b4(e=bU(bt`4C1}UEI6HkgT~yZHaOKG(!4QrA7=sBfb^GO zEff|tXK_|xWMwX5?{p{@(=Y= ziF#P2l;qLVHx5mx^PTb5b~CEOzgEqQtl$^8lk3&vcLZ;}M-j&ga^Ol0Qqp~kacN4) z7PQ{_`07=BI0gAWKg~af{jjTpiV%&T!X=fCr1(2V6=O|!!yL9dz7_LIa~Gskx*%KO z;N>U|f`XBb`FAYr80ZQ#<{9Tp_F8>g6i)2mXhJ@~7oeRsw}?EUjd}&Kv`zbiL|YN4 z6{|VTu8)?*T4h zquXRb?g6wU7}aXtlj2#92tQ(mCZq+b5!pdOZ=(6rmuKiw?L$d+sYNT$VluzIqlz|R z7fZoNXkhedeD{Yz=E4`O9y${O&T(mV_p!r#`O=DAteF!b@Bi{Mn85cn9*$+iI`wSU z+hx-|(?XPETHZ&t4Vth7-?*kgRfosgIhU9EWPj*y5{YtW1Ud~((Mqq^6KwnWMH=i< zTnn}_&`*pjiD5>Zq`fp(Yk8pzrO&Z&sIs3#^((Ve1@ktGHfq8ueVBNCj@2a|+RpfS zZDO?!e6G3KI%+TDJsKPF31A2IM^zR_wM$E2lTN*lBf_kkOv^+?8EF4X@klSeH}5yNlxY5mLnCe4p@qz&jyzzZR2MJ+LPHNo@mtb6SN+9OQI`RXkwavkI77hMa!@@~@a^w$K7I3kAz#MHaBHYoYBnYLu^ z6lsse`u$Orxg!n3!#ZiC+Fpk2FmV@+VcYVr{ci$dmhI(KMpyX?@svpKgJ-+A|F>iY z#)pe)mf5efS7_2m;P+dMM(GlKKoO`%ynp+Nnw{-mvZJ6*fy#DCa?_e1x|kpolCxi7%*6>`aKn6K?kV^xUxje0 z7xk7;gF!U9I5qqlqCA+Hks)+EUkU3+T~7NZew%26Mb8VD1!1d|oE+5hQDWAj zEy=3Sy0)7k4Xcy)Qi2gPx@y(7MmKx*_zAj#g0=y0`zwU_i0Z9LvZ zh!t`)G`8$ho~qro?F>Bv;@0fEs2wZl0tOqs?CqxZ^fNju$`t6%m3yXMY#qV{5;oH=5nW`M5Fyo87J|NO)=Y!)qv094!n7zci+>`tcYE! zHxJC*uyx~-RCMv7oG4b#1N*DS0@C4uV&D7s5tY2FNsrDvvBO_J(~Z4kosWuimGOa>utals~9Moa_v zc4CVMU}=69Vq9N}I=L-XKg|2{(B|5+6{72@nI2)AV0AN;PUU_c+e8!CzypN3Q3@8E z%Qs;}nAo~#uEx0qPPn5XY^5LHwy)@Wyn>sU5;taAoBK=1!Hw9-C_xw>zLu- z{*QuR(KHp3e9;RHhk=wA(LCk9qAAxQg8`w#_qHL{xi*TeeDaF_p5$%E9f#XQTeb^0 zEI$7xJwQ4CFo5qdK|1s9z0y5<$g2Q-(KECi>2xr~w8r!MbAwaT^{gv4dy9e*r2Q`J z5un9!ok=ew$@e;4Sbuj}#;)d-|300NG@oO-{-q*#bjtd+#yiZ3i*yTq1vDi$y3OWp zH2j{|x%9?4=zG!KQybyjIWg*-7yBy5h zN%g?*R;v97`1lCGxH7j2?ocf0xnB>H?yuJG2YK!e}?QN zJ+paR)ZVV6OuO^V9)6onzTULOP1q4%YmOCOu7BMwcGYg!`Usex8EALt)&(Yp_8x^~ zrEZ|CXijkQhDx^x&Q`rBXjmk=KY13R4HId0_x_!3;!1oTG$O2EZknOQ{g_nGOK@kf zk^czT)A$a4%k#OxibQP{|KF$LyveIx!M3Azt_Y)> zgO{zwJM3>v*K7;LPRE&>`rB<*&P%-vufL+tr`dXx*_#uS$FiSGW{8(#gg@ApZzS>u z@N?%cUK=%pSR73V#G)`4qi9%YuYd{!uvj1m8b;WvY*U;1F_eV~DSKIUh*&^O1_y{A z0s2=!mpU~4GT%?WM*z=@&&WXFc>X!2RuG8qn)^Fmmt9YCFoH`4frvV>MwSV zW6orQR!~agCXrjb=M+h(c>9{VHGNH^-eD^Zx43|5TrAInd?v`K1xaXE?s{8taOtBc zT(M}gF4oI6Jp1+k&XGV=kX82o3`cFD zvh{Gk{lSDZ%)*4J2m4B0)Ef=eUA!s~ zfe-GE`O9*nmcMwmZt{@0`}b-<_mES1{n@xN_A%VDQv*=rNk`-E$q=^LUx^Hw3CxO; z62e|3A=bIx=yjiwCT;vWSJg04X5lH3fW<~zE!n4DAA~D{W9WV?BAJy3+UFGicT;sj z%G?`w9PaY5tz7$rl>6%wP2C&)|79BeKQeiJhBV?HlO?RU*j--WSk&3&g~WJ_!v`Y{ zu2p_f(mr_voUfe}CH`ybtoTc=%YTNngPrBYBb~hRbip{($?z)% z`0vYu#2}*NE6p8Yd~UUNM6lR8LvgIEyYr?u#&rW=P>By*8Lk)Z%#mFNnoB} zUq@I@RDIZ~*yY##o8h<3yivzdb?FwZ#9m*64|3!kp%cvo-&DiE|FzGUDAGu%m^qCv zM$eP8^*cv1@+A6G-N~Zu27QSh{oZ(dWv@|nuoad@jk_z11{LI4;1>``{ULR`q&{4O zwzG5Pm|v@nUUodhzTP%nn|m^AJiUC}mRgv&yOB&3Y31zqn`7?2!~TQlF``0Gk;01jkP$B8TDG$(VV0}+JY3wZ#XUwM$r8Yg>nO4_FZ~2q=Z&RB_ zp-bT_?Q3RSFF3C7-dN9SJy>K%1Lw6#e-;R>!>@q< z#{4vTgrW5q8Xj#<$>deIp__VX>uN}OcvaL$S$6l7x9khb$)zNX?}-K4|El`DSA{8B z2g|w*a7-4pCmzJCkk%-+mD4nxMj%}*xC?cJ&vicUl6M383$8i!)DE47_Kyut5y>M# zyvxaJeq-VGyVh---TD%fWzyX0<-zxyx^7Kz$eq+j0Mx*3>^@&6T~M9#s{*b5M1!}= zE~K3Km3AP8XrbwpNz((ATvIlTdi^`V^5cI5S5q2xX(=z%H~3?*5hzd^Hv3jysBhR{ zc;DHELT@G>wa`_g>30`3;OU=I!Z&`#?W2|)G8J&(xhte62}3^u7M)+v!|Pnbbm)#8 zc^DbLljXK8ZmVKM0{>?BevPo{$l0isEpp1t43jF6C6J!M{Vxdv4;@B>h&sl@567zk zq|B`Xc9=}BLZZt5SbVYJWc>eilXo@sclf_fa!pA$eSW81_@rxD%&#z{2dq%J|3e3NCk8~jq}G&&KVCr9EaNsD|YdjU<$rTFAyzsFT{9((lD0a zbb6gV3$kDDgoZ$;xbh}q@rg?J69&tBt9K$ zmAp;kiEAxy=2ii(fdXGhe8|OGzVvU+zjH%^!wyNA2zA$%lY$XOr2EeMg#5DVkZKM* z9%#x65Z+lS1iXduqJ$**5^{^49e-8Z*=dAG&>ULrLe~BL9b#I3Rkpx%Gb2^+d>4sx z`O>#B^UzLvVJ5G>k$o7Y_{m$MoCJRM)KV^L3Pg_5Kcod(Ua?u&ZYSkVy>4Vb%)p7% zk)a7y8CTtD@@nrSui9xy zNy;tWzCO2HRWaSN-d3FRTYEuH%lWzDy}C{csCYPGo(ZgI$_U?!-I&M#Cg1p7DL`GM z56C(mkY%U7EOFzcM^O6VT}Ohi{_e2I-25yNG}&L?TJK**;FTgRrq8G6G3owGJI3K7 ze|g5j9PJTmKNdh69XIIum5+HI8xb@2a>FQsXTwgdgjH56z{r3F5VK&4d|nEtwC?%a zB+2`pX~WOon}wM}Y~`&iL&yI-r;hus#`y@~A6LX-Y?Y#SNJY z33K4$J_psJG*noMO4t>+C4OlMp3z%VQXi$G4)%X%R&Gvs@Bm|zO3AUW!SSBipCSSh z>*MU?xQpV;b9MaB8bA)Ga;>ZBj{pu(R zpw}1qqpUJ6G(?u9XVh*+5Jw>GsthG9q&IGHQe_k4u0t!0V=K(bI|UoMy*bU7w>4eS zwRD2RZVt(14J8lFRf zgEOm-Xr8NV>KLp_s`QAP=sAG$vghTZwT%vvW#=j>>9P)u4mTx^EQZB#xw$gtRlas^ zeQNl1(j#{=;rheNXX5Jw_n6c0CV zs^~-2rGXa_m!h9cd&|%h4tT--Xm&ZFCONr*F@H9=imzCPFsIW2y-jjNS?3g7O{4dG z!-V>;{^Ig(wQL#5ws5i~+>5m3v(}EP#T0LcJd9{SUaN(hwt3OUL1%|=)w0Bj^>pCa z4b4>5!R_D^aGGF51t-B*n7Kan?A4j<^xn)KVWc`79Kb|r{?ahrG12CLvG9H+5PP$} zzwHXAkF5U^&J?Ln#s(M zZ3WFNsi$7mDrY@ZKNcigi-}kqv7*@GGhz%2n0Fqo+?J}`nfLg2`!iV<+^N+wqQU}* zVl-zh(fU?@@T$AY+7FQp>XqtmkCicEW{+d!Kt3aIBkrp`&!!8%GksG~yzIId8ku;P!&vZVHUS*5xOPK2`db_rzv73;kVW9K6dr$qcRbQBNm4%zImag^>&)ldVh@FeK^Ea~4mqfu+N@kuYm*?snOd6yd}j{fM@f^$hrDNx8kCl|~wl zW_b^gy?54`n`*khk$7`}4qc7iJjaOoAxz|sPL~sktS38AJ_4!&*ucD?=PH5xW4a*I zN!bH^J9;jdG7e*68mU4!Z)~!ZnaAnpAfIK7unGq@AQPtlvQ-;$yD?TjVdPz(G;WCk zC8T?t0ZV|IJtzi*XA2V85qrB{<`1)|g$T}*nXR^Wv8l-(lxN{)o({)Ol03PHSJ22ghyVx*F_j)ExX{>Zi{K6GapR8z^`sun2jV9ba( zrRer7kz|H{L3#Q=5usjZt)O|qpckB}Hjs0-7xrrtiFS!T)QOmpSwOzUG=aXJWKIDiIOV5CQ-IA{AwM9RL6?7y!T)!N&vu0Jin|F9CGqXcc)G zUEj46q+u$nN(y^)-~>x?lu6KF`n`6#q2XtkB(!l}5-J|IN{@LFEle7;OU}Ui2NRp+ zWah-&mSzLGF)!)mskck<)6Dyjxi211(BnQ@Ac63bpGY`KtGuFGS<^#Mo9xxA3{5Hc z%2_{-D~@p=2EkuxVkc4wreI7$08=pF3w*RH3qYZ4db>Rqj74Sg191Dk`KROYV<5CZ zvk4A&@Tmmi>;X_PiN9d1O`ewT=J7f~cchzjMGOI?-P>Gruz3GGr_c?bD-@_8 zCc$G2yf_QEXINh?pgSXJHcE?|w>aW=%xG*AB#>5REtTbbnT7+=Cvs6WsbGUKT_`}V zH>L%E%*A^=&2BwE-lMi*$Zg=$J`6iVjhv~HQ-|u$pB;I{E&!pWbdjJI7Dg!(4#v!) zI+^n(4X+lK&7XmDM#Mv>SIK)uhG24lzPT(Wl{5wk*TpAUJ{4DikKiYyrWEobaB0ka z@WKv6A6yGcTYafuUp?M15yhm%0>REdfrI#zKT#CC5z;t;ACCG8kY0KY)_LA$KWA$P z*>p{0=q?q^RiV?(O}pY=zB}o>IP82_t?NN`6oCDVqRpR|{sH`A0x;-?a%vIautsFV z{1`ps?#zt)UVIxy9q(Oa2n(5Rkas$YKlD4F7ko{B!&E)HE$Hd=j#V1#Z^+u8Hy5vd z;bbMRSzB-X@z}GF(~1toDxA6f6uor%JvvL7kSGV(BPy-c_|>!J!Ql%+Mr&=IVNK_? zI8mYT2|D4_>1wJ1b>azXJs_cJyKot1+>}BmWmEE8woYy@ZWho&(KJuY-@0M>@CJH7 zw)EGiO`ma7qK7#n$wNT;`F*Ih$V&9^FqkjM6gtkQW=HRoj6fM zcd(4vll3iSx+d_^wY?hRcX@Dm>7-47`Fb38yFve#_KI|H{eYsJv`Gja7gN$wK>r?> zdLw-q#q*lZH6Y}K6c7mAzOJ1VC%3}{xE=%73mL+aZ_C0J+NxiX$MZc;C+*y&r0??6 zGFg@%n2-u5Wvw7>484*Ga+jh={%8JoeAnIZjXjO*%8k(t&5U2aE*=>Hv+ud%pHjo2 zQIz|tCuRudNNseW>I^G+a5pp94hn9b%&gh5%G-~kulp&U30%+jZFanTQoM+&-S^t} zeJ7VajTeepj7=JX?_{3!+>rb=1*AlABW|92run3J_I}Gn0E+vo7`(YQ4S=Og!UQl zqLZ(Sn~~fFF1M{2cRpt%$=8AVdupZVrtWj@7?O@m3+K1__o!>5rQHKxsN2&ys4LnD zT?_Z{46uX=FQH599l1}7XfZ+JQ!YL{QjYV8Vxoc<*fu&k+TS^fuc(eD|k&?#s0csG%WX3UUV__LmRJ+a<*T>bD)~CCmnRK1C(|>8FHRMWNm1c>cub<6Ive*wl=pR$U!<^|87DU zkBhip&|NLK+S2eK*2D4FB#C=~~%oBCKqteN6@u#Q-Ie zvu>aJvpggcIeU=VYF>|o8ZrwWhZ+@kl);_&h%4ubmX)ZaO zEtz~ex)Y6|{>H)e&)v0somqH9gg=?Iu!f~ouuZL=Y>l@mXrrKaW~cCd{W~6GsF{-^ zkf9BIDq-gTX?wAD@YX~byhWrgsn%}cEkMo9MyjTG!K&(`!R6<@e=Az|fw zoI%^y{c1u^O&>5W*^n=|YDoL`7>6-u@*GRgve3gVKqp*$quSgigmF%}E%X{HG?9!V z_1mr4_FZX4f7vFq{fPoYxrIzm%{vor3tDdLkDn~Kj;{o_PCvrI%co$KAm2~NR%%Z! zleWdM`^YU@p5Bn<{y)XdFB`rT{sDY1EvhXw%}}hq%}A9{Q(XPUF(#&YWX_o+lMTRU zrkV$=wr_nJSA|+dI4I2sr$?Iw=R=uj^>93SCXX!ns>VEdA0K-2EHW4v7(nogRWNEW z)5cBz6~njZ;^8V7&zK3-$zPU29FrAsq}SK~5ETMOjtVGa>0DGmP#J>6DI}~mDr+gF zSFlLkxnl1_C>=e+MwiD(bvMV=K)14Xx!~p>Tx{*&6?PRvy`Q$W%?C)IP58+*oU~<9wIM)T69!ah!PpV;~XFPi(}aTtRP<15(_A`Am=~ z*S#{i#)~e}{ga3sG^mFgy*`Gri96nIx!cx8z?PGFx{59xmbd&^H=F67ijMud6eNCe z=Z;0J++5G_@dve+N#4G@fJ zrBwh6gP(|xdtGd?0iANEhag$+r|#2fm#&mwDYL8TDDAQ=E`_?HE4gzt^rn?-H`vqy zRD$^FL68WYL$~&$T>?WXXzTpZekIE=VPWt}^DUoup#89T`z?B7tZq~6HD+Dq0VUQf z)3EA{7B>M!(U*x+C%?zzx`G+;(akI_mvN&m11uuWhR}K*HWB>NncJ2W%Znkk8+gYf zEl;ewv`XhEBAKH>(Ju|cp1`M*FcP3YdOHdD__Hn_wY997G5GRtuXz^HT|6~)JaZ*^ z&De9!x-!UTKvN1HTn!+m6Iz%w-Kv@$R;a&hqDrbbXke60ab4G+g8LrqiaxWJ5U7a% z;Fq7F;3k+gr@*a7@m0-f#Ry|EuAu^HCjpNG!+ClhwxZBetHUIWOH!;;Q~QmxOfSx% zvCo}y$v)4qwddN>5Itv`de!{KF25{w*8KZ)Mz4q!$yw(yANeGXMwDHDn0^dap^v&) zr5AGTBWe|is)B;9oX0c$I~_Y7`f2Mo$pH9=s#R?k(s&WF_`U9bwZbipa)IeM&|uH^ zKeFo^5e`ZiG2bm3Yz(iC-p)x>jyfNH@bAdwLToY-Wl!Eqx{SqZ>+LFLDf(zocu#C zp(e^uTAp(>+)0ck;YxxF)WuQm;G98pWs$#ozmigST*sKNLAgKvscvy@H%dZTqA@i6 zU9a?y5utE9HfO3CRY48_2JiP={p?v68BoV)D(RZek1i1mnz?LU3>Jz=fb5Q5U|Vj4 z*7G-`9Md4Z$~4>}gZIlHkz1z@=W-4&T9)FLYz*NSf4lBc7fGm{d-Mef2?+@-@Vv{3 zm!pc)%@4oN2^+TPUZ%lQJsr5M$DR1F$*gdc(oPK6wdbcMi3+PWCFQV_yZ@A4?oLgX z7VcE28)JqJk#VA1n&fC}m?d65uD$MmK>DtuS`nZ_jfBq~uclpa!3X}PKDjP+)8}bS z@{RzSYI&h)dCXoehpTbOT-p2Jwa_f0%7+F6T~abUcXO-gY?koP06UH>BPs*#B}m9z zc|E3=iRmo47JXS;f7&-E&73>c>SlXh=RqS5;nXD{$QUi}Q0829ptOvdX+{sQM|Zqp^qwphoP<81TOKrb z9)CB(8~(SMH?QCfP?Ukje(kp0F`{L%Zsou0+Mtzn@WBLz2A-F6yc1e%eRr5L!PU|z4$RY4G+bp%q2N}f9BbpwS1Id z<6Qwc1o@{B-%0tQ|K@f0Fi8DCK2gG+j)X}v=k4{>J=6qkA+MD2shPD!BD8-A(JZ`7 z3#|p%*;E!@Ee*8NHm-oYjE8wRz4`-RN6#{Gyi_y)cN}3wcdAKz|Hg1D9ujfw)vHb! z?$>?;A}lt~VA8wT3~0EUKL?}g(33rvGL++e?X2V1gCGbX8}gawNYt!wI){1&in$j? z(tV}QtfG-E5~;h}S^IHr7K?;34TqCgwRGkyzg?!nT?<8&ad#_)=E@w(#t{+cd^-I4 zz9(;A&BdW}t=YGmyOWFjztCNXNSyML)Ds>(u9Yxwo@u+8%3w7KoxISc;S?gZItl(c z4kxfSwP)F6rY>RsP9PmDON&v*^|^No+e3T_`^3mfSJZPYzZ}=q$B37eSrx0P3*BEs z1v4%cg_iTlj~efyOtzLdR@(^y`u?kL_0~L3=Ix=+-x8UEIE(qfZKEyb$(*Hpf@2c{12vrF)rm z8|ScpMx;&aP#H&6D)3X=rm4T$?0tR|iyv87t!f9b)03`H3F9~$V~$WxkeIJg-1*jQ z>-hC_D8TthZgm&2d54O31}|?bbT4h9U22gI8;Wz-PhS(-mR-y4YqELrdR)dkJ1Q?v z*U>d9!gu>bUFS9T1yY;cv+(}fu5;&mw6Tl!F-{GJT^WGf1Y?G1YA_+6*@J1JG(9|i zA&;TpT^#AacRqzf+v{CkKGwfpG!1JLf`u2^Gjt`rk1UsPoR2rpHY>x^EdeIq+Xn$! zzY5|q6i-IV!#Y;;5r4bzQqecpW(?DrdTpfKQT-jGcACjCV%k5`g6XMC)HN=dLL=Q( za(?a68fC&(kbe^l%pqA}Vk{!;Ra1S%R}{uQX)<4Rl$vDCWBw4nGF)iYvN! zj>Q-g1;V3}7*dTR~2{@&hy|CY^y4B=}AT-{$CIu?UM!nB?} zs%sxy$Mw`rjH&lQmM-kFl5j`o@IKn}dJ9JC{Zf&_SGD;eCbtye5w{ri4e_{g>)-AH zpG(|D2X;S3-fO0oX}un9;=E9mJqZhD)xZyl`Zxl$O-fWzGk#s-`Da@$^)zQJg&HC} z(ehSygp2!wA&grt7x(7$Ex!j;Vf`4gc_95mEkOH;{+++pGVQQhWAY&qjT`q|AoLpy z$+J4r6{7GK*3IJ1?M8066)gN~MA5`WD!B8<{b?NO-6=Tdy1zU6*tzldt!3F;g0e!= zu2LF4=~AW&X6A(%qh{+LTPK&RO%mcfxQD-WYaHlp677h{&Dv_~pd5-wQEa+WE1kuv%Y`WykBcY7{IYl1C=u5ioJ+$0?3T0Fy+4`#9IKI; ziAE!y!FG*Luth^Bl;o={bcq8S1Jfn^o1C^oYA` z>u2(ROLa*uCS-jZV`v2EHsOphVvGXdua`(&`ZruOe9SP!%z#qXf-A?d=G7GBsNtQQ zq_J)yeVmb_&Ot1O1}U4i(FX0ma=Uky0{X?ZRG!tztyZTp%kgXQhO?_cmjJM03pi(c zSLklK)8GyT_Bge5yh@lZ;CY(-^47mw%lPxSncHpnMFCx$lVpefx?*2jw4_O*%D(AfXkYB>L_$wegaPexKBao$I1_ ztX+2U{lWC*T=^rW+T(6dofg|Ss%1NlGSy?~nsrvz$frWB2ps_bQ8&6bY;?0XZ}y_f zC0{bI#ULL@sFz2ap%TQA_RNORcE!-!I+w5AK!WWd&+vcqMf8uBT#LUdGP{hyKta%C zYU;@H?l0=X4lJNdU&_MqBDsR|IknJER~>}LeES`(RgI*0LH_ht;M1ihx&yt@(;a|D zH9%Ixu0j&tLfpS6oo)1-=ZY6-+$+hEY41`78ZY1X@&I0T`Ii>+5zg{#e)t_KBmCh@ zQSbLerLeHBf6GZ9e>OI<0vop==5pHg8aTS2d)r*OSifve+}2;;dhTD9(U1R^qQ)5` zn*^RT7l7Onhx!;)F%HRaIIB_W(jovDZd1zG`#rK=uspD`3QFJ^M9bpXg-sF=Ic znw;gvOck(<>M(673;YPkHE7;%md>gGsAib@mPkja8&M|B6;T$Cm#l&3*N6g!y&lxj z=w}!mP!jR;@JVe@nGZnpFVfoGxs;Xd&)8R@u^hw90D%8^ERI%4bA{R;HO#k~PgfGQ zV(U3436WFMoGw)Itelg94uDNOz25_5+J^cHhLVz^$jy5oA`=YHgdv`;=b{v}%S0QS z0j$xS?AxQh5>kGTrFTKMX>s-qD4)ROo9dZNJGZ~BQ_WS&__`E?R`Cwzn5@!)`K%=} z`9IGeaBg_r^#}nOv zmam7t4fRMHZ6A1dYZuT8+S7HJ?L^1JU1f8Q7G4fgRe}%gXWICEI7fI-W=_Fc**4|iOxz?)e>m^=D#`;;~hc(R7r^hWpGzG^cieTxNQCDd~eD5N&G#AC+CS7lYl zxTv1Oety+|>C2-MS?H&o5ogmTtSsnj!dLj2f5s8_X794Br8s(qvT&c0wWvU;Al+1c zSO3hRqRq4l!pS0U&fYKIAaLSwoYHTdMR%HXk_lq~g2cF1APxa%2@|y9{jSbVNfL0Cj}90&rMXwsp1?FN|V+*ME+&{vA%l~ zEc?;Blt5S1i=c`Wvu0<4DPbpw4y&C<_Mv?#X!SSb(6zLS13~;s(vF&kCh^W6-6KO5 zIP1TC9knfE7Jfm*QM|~1&P#ViYiKRkw^!UNeA69B*4K^vui*cysUNxBVp$52T(B?0k8D2k4{FsIz?=ZT-cULHVn^ z+3TaJMOvnj)or`2w8!pd;jZJCQN2s99Fzphx|)>3@_$Q^VO^K3D!C7Ay4B2=%;9UI za)1v}OcUzzUM6uPt^%k=^dpZH;t9__JD(>*XmW*$jb)3}A(OTF-ab3%qBjbNhyqu{ z!I?CmE?!aJXeo%>0fk|N$)&_b({vd{t|HzB%p zBr(>Tn*BJEYa^=7B*9oJPXuy49@|EzG415Cm;Co+B+D%@i@nMBz09caa34F6AnKJt z6zZ3t)#i$pDa`L=1nliB(c2i>lyYYqa&5e6KIwuX7bB;Ye$XBdZi4T zr5*i4sK9rgNYryU&=0Y%mWv)n*tQ-kJUw+ON}TgDLYwcqPEm|K-Da<|?G|wPYb9ro z2USC{))q+RKQ-pj$~}ugxh-^HYF$O zFYZZF@SUzIL!VBinueMVEn(Vv>6k-QEkg}t(b3X@dKnYE^1MuO1613QVHO!*fe%Oj z4uOHUzCvJqb+z$v&~MFe(?9xuSt`|YmJ}(4JZxGtis=^bIq)ke)-W!mU+o!q zMAz=p)~)!qA0QwV79&io`qUv>r4=N14HiQg15wgi<6ru#d|9Zrvub3++;^J^9U6OQ zkv^y`Gh`K$nDNX0f4?}!nNogMT3nGgyyF7piTZf%f0LqnzyHDV^=2LW{a{4MI2?}B%s1pg;V3Q3ftjksVMKtxlIPsS z-vqumb=vdI0T`4DSJy>L3#(Bz3rO*!Va*R72xKD^>2%`jm59C|41N__i(#k3OG^vd zl?%4ZF!)G5MP4~rEF~ui40Kw^jeX7mUH-!)TTz5@5gbaxOoBx!FmUrT*rqXZWbcJq zWcD|VgNi#2EN_nhQ(3mcshLH+n_2un?cDYoAJIM%-84y0mEVSe;v_Z3M+J@#z|$#E z=ZaJ3)jmA(39kRp?+93ESp??5GJ}9PB!q2U z#$Kb9(I@{sa=_KVgXS&f(+OW~0f7Jba@@*o*D+Vw8s8r`g-chg50Aw`9kpX6MPhe9 zmpC{?c_7>NqBMJmT#aX7k*R2fKpcww>P*s3pCw2W+rT5?^KZzS@N_QTJOHGsO?pVE zhmlpfPml9?oWNTA%-FMta62F$d}Fq&>tEFUJ@|IF9|pShMV+i}{m{Vk5oj~E=L{)& zl^gQm4+gF^AjDUh2-1tGPb#7-?UPJ~!$v;v&jwd9UZs zfX)1i%b-b0GS%a1I9ADrxS>RjX0O;QnU{}kw;160eV?=ai$0!GfT_u0*XJNH1`|J_;E?vWbl9K}U%z-N*Vv$P4>QBV#bw zIVw3MVaOYL+0nAI##$2I+CC4SHZjt|0sQI#c_%$goUyzv58lU}67img$@#{4vn8Vi zv1&Ff5;AQ4o^6*Kel3+|wdZ1UKr^iTVA9qe^CNfkqMMXQo-g?i;4uBjsrggPz2+Ky zXI!yDTnU?{2yyTkO%mHPj=#$x5r9TiiFMKLY6|7cDH}L8ehb;&bmKrOmuf{F8Gxk% zPl6N>A%tk(ix6aI=ak>>D|N^OXjuomta(vb^CD~O$~Y!Xz#<{~El=0SrUcY^5NMai zi??NqL|7gt{**D{9=uitr|OVN&$-$+-Xbx1gKd1w*seIyPJsKE$cJgU%wu=MP_7mc z_WSlg47&l!Z`QJfzC=;yxG=eQMLVDcpJZ5;<9>sxw=dVeI(4ueEBp+u6S6^aya%6U zXg6Q_G~ME{Ys-x&UBu1X^Nrc27J0}WD?Ar`sKX}G3V{$7-(Uks-4@~`VNaBZ*zN!Y zSwD>j{m>LnEKHn{G9ghq;`I)^k|*f!bbem01|=+V@cGx==XWMTWVc=%f%v3YInp`w zS2XJ1P+Lg<^O#pN)R_-K+b-fwz(wDo)mlE%+cI7bNgvMO$Z^vT%QUZOraUN#iRUh7 zcpc`XA)XPb&&_%10udLZcKr1!sEfmS5~;@<9uyS)o<1?+cSwxASB^0OBvx8=f+>_m zWv;(pe0vmSX~;|cOEyiJhuHT(^o`%5M%s{5BaO(ldBhJkYeWM6s(_3g%7Wm%hsFBW}1X%9hk6{z|nV;+Ur$ zY8$>+8`H;8Oio*MC3BAHtP#3qHQn$zAWX5wydJ&is+LElUtA&aP2il$enNedw~JF+2I z{@PY3iq?wc%{sE+xZ1lvr5Y^lb9UHd0=`0Dvwnv4`nsSm{{FofuVa+*_eMQb>vAT% zC;w=Y(Ky=0YWnVndUkOFk*Bh1g2XSCo7g_NXI5!TXp@MhZFsjDqGXv=!*rC~iqqPw=BH_|7)2Ax#5;$rc^! zL4dI(mtB6|jX3KHxc`_hwMv_WKe6tF(N8hO??u0vMTMznKVvJaOqUdCpRs00k=wZH zt7%^>apy47U5F7V$m$%LYI&z5-nKb-c7LnCU)2f(m*kDnx8LKfRU#>8Zi9Y)>P!S} z>o1&2%+)og=b+V9 z9kjf~jB07SVabl--fO{yAD=1PT!5Meu z!6<19S|{rUnEaUVc^T!G0D8^j&~>oJ*giSj&R7T2!8w~W<`LVNA415ba%n&RLwM>W z#V?yrELUcj65-BwEtbQhIbDu;ZS0>T<`~VxD~-u#z;bx%90c8pQhhlrts5b0PxQUc zvM%<+kmH3-@2fCYKi7RGud}H<7WC0iJ$v=1R+@$lQH}@tDO0GgGIQkaYip~G@ zxeB}A@gS~MS=Sy+RY_S6UAuj14m*vnGg;sj#r(<%O9K;ZZ)FtwXE&`Uuqh%Skg@dE zPjd>|bPV+L1-b959yL)X)2N3e^okMft2m)kFABk-@HI7jA+sGg>-d`r2j7vP8sx`i zUwYA0Aul8R<9=&{5EUmHS>AU=kmoqF5=KQT(8OLT`v@d)YIY$ z{7~#x{wEljQo~&xc1M|!h;a~o_uVwQ*^8X?t(f@FLL%Y9j$n@D7Y+TFY{6o9h4#Y$ zr5N185)v&R8!2-9F*iWd(NO$vM{qX<>cR*h=sylPfX<@1k0}LExGF!%*WH4xJP~@z zH%p2C#h(X~`^5+QO$}(@22~?2{fPigCscrqqH^m_;P-)QGa8N1#ACE{lL+No%0-dN z3Ap+|1MUcWbIZ-HW>iz+fUrYJZsGlje_9cWSw08Xz~Jp!kJI|(DCnzQtfpn zY+0@EefZUs?BxR~k3X{eMs&xo!v|s;LLo--WyE7zm00qWdVUI6?UiJ+%5HEG4brkn zmZcmsgLWPx@_B~z-`MvwN(^pz&DhSK!(b&hr1LHKe!Ge$^L&t-_4M?d!L1Vqg%(?T z@!6)idBl1d_4M_z|L)0F{Jg`_JK(Y402klbnCq^g=I-2x;&#Os+65lkqYa&eaFl5akXlqtt7yO0F=dXB`Rll5zGT<^-@KkD7xdNqd7zPA`u!%` zp$`u3ZJ&#RUU)F6Y<4-xrr-X-6usGJY)h8nbk0W8XL?Sl)5>=<=nGyAx(|(Nbz-R3e-R>N=9v%dmRO8gDC`{ak9t6UZuP}f8d20j1YMNh zz23OfT=tzMKJ#hWArTJs@p^y2bSa1{W^zu&-lY?uz3j6@;Z+mqXmBaZ$sQdlzv|C` zkvM#x03}KJVXYHDnrvl*ZKUGrie<(1@+E=v`4WNamjHZK?o)eXsQfn73>`sMY>Kxr zn?uycheIPmN&l*&+=KSj4D7 z&nF&Z;}R5Os_F9%77a_3=Py*ygB3sAW$+bRA!W9|U3liXkVebPbxn}lEQATvPzs1k z2_KqwSC1!K6SMwZ{P6!O_xvUNKK@(kF^;!W>f}ZEWx6cxD;a6&)5Aa8A$|FlDpL4M~)*f zjx%XKAuzPX^6 z-^Wpa7vXmF2vtduHAfxsUxRdlYV0izA}xoizIF09%gyoPoreso3>%<|X1~5+O6g*e z$j_cJAzfj~ z6_aSqYqsXw51-^ymowVUt;?e|pw%_eSqpL&PN1Miku){j&QMod_jvZD^N4CP={D-% zj>AigakL$KiS2!~S^WKwT5T-+($(G8?Mh>J^x20Ifo5^}XsQMK?xK2BNmX%lLX&j~ zb`2mhECic?Ib5rXJfy}=UO-iYyZ8Ghl=0-8h;be0|8Vj&dbo8SbVuusBi=j_)9A!a zt53zKYoX6-?C_CR*}kU!*BpkS#mpKUl*9mA;kp*jbUE;_Vc_|RZQ?rZr4g$CtW=#) zy^Qx_+d^#Yg`6dK9lg!1V*`u#?CQ;v8L|<5l!V$zGD(?u<)Oz`e(sIJ(H&=^09eNp zG4ozqyPThObpR2=VEp2_F9FNXjDq$l@IN;>0@hpwu}Z)C>WqH+piI=C>TjD?9pWaK zOlsm+mH@aZdm~Uhw^8G>A}C>@hUleF9a`J;-*A83iIWsN80XE4S|* znhs~szr|O7iH|95jx&_}g3NJ2TS`4%e^Ek3JK(raUAD6EjV0IZ+tpQ3Sh)!5b+063yj@Tn~ z4h>9bPANyxz%$vSn`}=xK-`#0>@*&bc-Xph>0mRY(DFvVQ707k;OBTcRDhRWai{cX z+yr#uk~tJ(_~sm@x^{@z9MRj#G`mWi3r|R(tr502+H>JT!LGV!SRH~0nm$R_G=q)K zJqIO$Bi|^y=wQm|_$k|Os^dXwaOjfIPCfiz z3!qRLY*(g6Yt=$dQgX${EWWHR6P}{SC@C6Mr@@aZKqJAAH({^+AKv+$Z_ZllPX5%l z&a&~+t1B8x9 zX|RD~e67qr#WteOHm)u6VwDp3aH$D)cl9+`Y_89cvDs~Wcez=&@Wo!wSW8cD(dcuA zO?Ekp&hjnOG;YbiB=A)#7~z6;De3T$#}uGDI|l~dTwIvfy(1jhx4(JfTm?1Mz~Tj^ z!oO1Lr?b7d5+L&W(boB@-NzRFukq+`)k#WH*s!vxaDR1EtCZ+yIvTr+2oLg=v|yM@E7afcD(LmS#gzV1fhW;# z#odb`z+9M^{CCDTAtkXcT)tW9brt{YHT7zh-SZgvSC`+~QW|Txyl7Nr&;3EhSY_Yj z!c=;>15$Dub<*^sPnw+ShSR3uVBbS47nx_tk!0UJDxJ zqJ3cKjTT@xT5hMtxgTl|c8n#5KmERYv2vCXPTYk{6nIo!F1rlJEuQV#)BG&R?9J1C zYSg**7M*k#b;niad~EeaBK{Qo>)PWwZ7V`_iQ<@u0>1zFw-?oMG5e`^BMS?J z&JWX-X!d|YHgJ^uvktES-`DT$6Dg4jZm1>*b*|cs#>sYqO#CmYmeoH04$C zNuB6n{kyPMo36x-^VVG#TA*h8GY!b0k zkAdP-!`|<^z3A)2rmRX3ji11U<4|;6wu`)K`{1710J}CB%lNO-$bmn!#D}XlP&bCD z{UHAD;m>Jx$l0{ZHrkn+AhEuN-&ABu$K=~(-ydSO>3$cAZYCrnOJxgAryo+uZbyno z`HTGcDI_8OBYn}0`dtbibfBCXs=JC~u#27h&{wzfZZw!kmkg72W#QXREB7wic>Ilv z1Ndc5tBCDk3f?fKR%QJ=a$U)Kh`+XGc=-O{A$a!JgnM@UoP`4sL6wMOm5sfk`oKVB zf^o*akP7%=yWr{JCvXD_>n*1)!^3E%@5N{dumQ}9YshZCx&iExbaO8*cegRf-KkY zwL%2a5tGtSx28|~fL@DwcUs;j(Gf{I&XX|UAM-RYVF}s|G=U(sx^7vT5n&PaysPFS zJ2!>j<4-Ce3JNdOo=@W&AWg5M@N+aPAdoe}t^D|g3WC+EF6E)op*vl#*~1vm@%t7d z*21N=0%IdPV+6WC0y5B#C>g-0AvhW8AA*qG@F1<<1Wyx{lv-4TH%%Y%U_F{Yf`W!REsagaX~#S z0E%(&kGG2q(C!?lKIf+1c?#-UsRwP-+}o)yAlfsq4xRbFv?scl3|Bet)*FwJblgS> z0&4g|>D$vmR#xEJxQ4gMq?lI9VzN|jS-{-<=~s^rhj0-OB`aD|rtW`UbHo%lUC~&O zM~Oqj9PaJD1-HvqRC^@y!Y~^5Ub&~NIx-~=7JMY}trbm6Iy%wL4_og_Q0)sbFp|cM& zWCeozowZ_baXp2RC^BFh^j%yoU;c&dTa{OF5z*F$ri-S*Nm8ht0_18znVeSLsVQW%;BsmQs^zbsHmuUIEa5Dp>fD4n%85W@*_T> zjR|8{^%z9%*!&%Pe{YvKLQ`|hm4=^ksAOO^|L2Lg=cGPC;>z!?8W8`I(NbK^WabL! zHuBD~-tS@Qs?~cGCt(y;e2;*x;WeKMJDRl&cr3q_0~^CZ`+~r&Auew;l{i?%fhoG6!(+-?YakNkua=*c9 z=swrt_-X>+Kb`pnb-Q)8eBh}~<~h~vCPR12Gam&SFCjB!6FXkQ+7mw;k6X2-FnXVm zP(F?^8Nd~A9ql!5D%h8kL%^?=&7`;> z#UiN=*{a?9SAqLOk&T5hEZS6mqIm@km@K}t(aOzxU!hNTQ%&{*j8}PH2)K1(7V`As zV@hM{Y5Qt1kqJS_4Zc?FPaRsPB}D!B5PV;X(U;Ly-KjQL=EME_GN z-b5H-WU>y?a2Pt~V1`j{jmrw$kE)huQ(=emU_IDdAC`;Ohezi>> zV|b^NN~l$1fl|`v9TcBGDkZeeLAnmZ4vU|#!m9G8gIdi;?Yw0DjD@%{khi`gJp_A{ z{1-27H#R0|t@5Hl@S1?(9<$S%iq@A{_Ki#D&VEx%Yt&FwM(TIC+e+dJc=)uZ@-ktu zDd9G%MiOkimZ$9uQb)|TJ zI`632dIO!SNJxmPF963Bz(($e?u<|T7jCazl*#`MU435xz143t_E9GlT>^`-_j)nF zzHx$~`V8HlP=%L9cTH@4mwPDI-#b@QQe>n|6po|0er4OK@Wa(kf7@vT-%C@PkF-6g z;sKuJi7r`i`7QVHd94L;++%x7?iqQs6j5nzNcxXa-E{cs7ypaJ)u zp60e?Kcd#rQE==-0&b&hue{f8`XsOB-J7rl$(;20!IR&*7+2?C3QmN!dcYn-BZimx;p_eQ7=rHtgYo%TfSrq z+VbR~c3poA+NR0rmS0Tn+DKW$R_p$IT>^O37OxwKiYPBN$ z2wjz!@F<$(X{B>BOC+~<4>j#{YdZK_{yFPXd@W!mlV*mAO(%k~a)qOQ!RxI;jHqJ2 z6F+r%j0OISXl8QV5ySVfIw9wQ=l~%6Q1gz2c*=G^ZFMg{gaZQkDwG(1-`t!L9nZH! zW0mzoS3%*eRtl?83B=Kcg%;;pktp1W0as!8j5q;yo* zUh2(~m`ozRySx7g+5VxN{@gw3-`BIoZx77LA>Q||Y7l1f%X1m01N#|w1AA4%O=fYE z=nFn<0B@Pk%b<< z7Qa@Bkcz=e<%mu_oOs&AAt7;2n?K$LON7Dy2=YIJZ1t%Ov1X!Y4q}!7I!<`5_BeDF zV$Pbw>exC(A2AME?z<{42Z`)ey+Q8Xf|D8voKI`4Pc)58&MH_5_~sGQ0m@57g^Eep>87No=SlgBdPxK_}%WE-kcvkaYEIQxw>HfC(y z0g6-phYS7m@d4^&g^}hrCwq`$?Q$|aB}H2=ggosBk&Ftaeav^ADO*k3H}i*$^Q)V_ zKp%p7mYGQ1gNUkzQc8-sH8HcgesWkThQ>6StLdL)6=%(%nL7pnS9ltLot>0AG&ov? zAXpi{JA*Vu;u*_$adk#xaqgb|)g|6N2}u%hL2e-O;$~EZkB72XEyeshn|%iD@yk~Z z;ZEGF1As%e44-->GmP-x9MpI^%{)g8AiP9yZOPk#m(Csa$i0laaYDj{c+PB#_5*`o zA$Date10ru8RThlMX3@$o;NJe$T0~>yc0uW zR?0F<2JD{@#oD^;>_S7Lc%iCVDWsjv?>Q*-g88GP6D%SlM>l)L&%1?zK(1KJtTM>J zkOcU99gnifK|0xTTxD(Y6MBE|FXaEY`l_h7nr_<$8h3&P2s9dk1PPYl8r(wT?(PJ4 zx8Uv$jeC&BC1~RWx8M*Ujq4rx&ON`J{ZOMG_S2F%*P69z6tPhv9ixq7!mu_0Hg$mB-xF`E`sxFL$X?5j@vcJR`qjnAB>$dLhv~n%OwwEr_s3>92mEau`*4T z0IS*di89gGPZGPaB8-Pcs2N7Q{Jmba*!{SBv6D37aWJR>9!&e83@7$7ldxh%hmjm- zBLZW;zs|4pbcrMrPpMcEShIP;G5ab z#(UKdOQyCtHj*dECRnDXt|jIav$_Yj#!b zBDs40O=A6AcdNY$_1GyU+P%`ofC0UC(Rf5%n>Y>01;^Lv_b{!n*`F;7F#7fz{%Iu+5z7GcE#Uai;)->kHQ+Kq3GcpWwETbMT3Fi1>I26>|9guuQ_q(S4?q;%hJ?sMxX7F7ujHICh|VOH+^uy-)q z#GeJG58*pcvxl40jShLaRC#Q^DsuJbHFS*@ReU?vND)1i5Y*Q{o9BjP7eq83HJBeM zBIT!b>Ft%DQGyqU+T13NW?Bb?Q#9MAef3p*-01mbl|wnsrTf!pLnpNyt78u@}nbR2ajE`djuo;1( zWO#-3L?ikocaLiw?!r%(vurPiB$6t0MFlm;;s+S@NxA&M!ydBk!cwvVvEX`hPRf(B zxV)VCviYFsK88|)zy(K~(Q@mSR50lb5UE6}2+6u{Wyb+}wXnjf&(<>vOux!tt)}P0 z@(ExHq6xw4(;m@c{SmNcDZq1R^xF9=?HMu#Iz27BfF3b57l31Al1S?8|0D%6fC|}| z-q?;^QSa4<5(r6!97Wo^kEf8}i)?hr_01mvWM<&F@^o`18HXAWo3=?*T87+@WAOKU z@fU>`ln?%9j~U6DGDyjR`Klju`VGY+>uraZKuf}_-p-vKUJZO%sb+2%p>E0hNnGUb zKC+eK{6cD@OOt}B)<%iS6?sp6J=4r{pwQdx?0WP3ZoqPm+SPYdv>3!`%L<-n;WmoD zP5KPKYoNr2Z#l-7L~_VW6pl&&B1fIue9kv>sGe?z`?oB$z_wN8WH!NRD7p+Z<}C}K z_A@a;=9KgEkn&@JxW+W>9}=dPU(%NzyaXux&K~lfPB$LscO1G$8VM^g%purm&;*Nh zM|0=h&Co(Mv}J+MSzd_}RYw`89?4X1VyJ{NSc7EXrtBGW(sK1YuY+{kSST6;&%L}D zr4Jg$9+a1K9H6J|_~=;qF+DaYP4RY2XiV|+>W`ogW6Fi;^mxrYGCh%Q->SJyW1BIk zX(9*^nGKNhk)<+-=G@)>I#cK0%P&6#)?_tJ4Wi+?nt6*i^ol{%&7Tb9m6~)cmaG zGqgnwhW0;QOpk0|>@@B){uw!(HExf;m;a2qWXR?cJirhnA%iT{Ba6M0q1HS-le*9D zs%JFrRZ$@V(O6HTmDy-r5-z|IV~QaGSbCrj#T(IMgaNfbjBGvADO5GfD<_0i3_)lL z;e&tJ{2q(qgzuWlWzWQO%?5oENc+bT!C-U6L^>G)6PCaj4nTy*gewaE*62oUe7*)M zHCQj4%k%8%M(FO>Y*`#7YpP^!=|lpKq{s+<02@?!_`GdcCZ23a1fgPi-%6VKuSG5NJJtuMnP(XeNC9u)S~ge`|;1pQbR6LoMNLJmrI_YU<2t&J{3eC^f@Bm4Gc=@|;WfOG?c!cL&vZ`N*EaC_I)W+aD zj8UhLHv(HHv7I`m*4Xrl^BVUdQXfUXOXXGa-896Rinc6mynnC(z z3kIxB&D%`EwY>;n^uBRP6|B!iFrk-T*vpvzM?ue2i(-wnWOc|+`^ehWQN?IwZrnkN zr{l$DN_%t^Z>kYv$g<{#*XHiHZ-rm1)nSUIJTg-4nv5PoO2mxT-$T@1i?Oli(<;XO zgY)*zP|Kj7?g*A_RW#%A)|qWH2UYJe3k|mf)oct<)ye_YAOQK5qzKx1fEZg89-?-^ zV!-eJ2B~NsB-v!X@-Py%XYV)+#vh3so~Qe9d8~d{S;GcCXYZ4OjL6i~5PJrUN^E=( z0uk7TXKO@Nh8R$qtds4VW*7!YSQ}8=3G~hFJ@O$E5|D9OU?9pBS})n?42v-7H`GLi zyVj3&`F57X5s#eP1cHhC1+uWNS67U6O4KKmcI=_o}|hy;iPplr(GizSIkkd}|hGDo*- zQLt1YuX+fCt?AsyoYYk%rU=o)YDlMmb;-W?eg7vzG+&V_MFzc=g!MrgYkVO_pH_lKX|S((Z0rF)g}-bs*>AWm&0;7^fqwn#l8s+qW!ro1=9 z)EAY@6p_bKMvFv53}p`n7-$TUi7TtjCnnn(mTf&fBtf0L*cJcYOUPBoqkyGtu@FB0 z=Fdah%@0&&`UO-^MlkaT5nill|qSak;wnCkiAfr=bn0a#7RNjl2A$|W#ZD|bPy zSUZ>ou(Cn{%60*>G2`zlW0umtj_7Ye!gtV(G1UG;Td8pn;g~Z!h4#LAbcR_PU8GCR zJ4rOW4-Tcm$C*8y=EAkUxnigAr>Db1nbZEQ>70(6-!CT?!th@H*>Oy5VIRFeE)zNW z91x8g6h`O1DkYE0&@n*lZ8I5zO#~S4Xuy0Yu)~Gj$HPb6=rATG9I4qPO))h?sfrd^ zl#ER+C@2~D%e&9uZ+#_)!}1Ebs}e!alQI4it%GCdABFk- zFV9CtaJz2uYe)=#$*8~IuW%!rT3st6g2CO}B*+IX8EB$zZ^=&jD^>ErY?891oro3O zmS~f7$|{n5RkCD=NZCI{uIS2Ev9G0;rk@WC9quQJW-Vq9lN?^Am!GDe`FEh+C-N$C zyTPh70vMx(ARb$+dg}Och%^#XVhw^NSS01}G~t&ScxBG2Qx&3ujxm8M16D-!RISVq z^5Svadd`X5!8>wxK3iW}@;vR&>2UA%ysf+5GI-uK_{-vveeEIX!R2l>kRu3$!TS~G zs@j{*fGn`tQ<7Z5h17$tG&c~ipo&Hi6c3@Elqw3xrBWP9+N_j3e|hYFCTUzc*?8%A zQQBq+XlKH`z7f-#u$aBDR8EmB;&`%lLpIPay$};IUsIzAIBK7q;ks?V?jeRpu$;QZLH(?zU-$;>^bWFQKLny;p)CUKqNXCL*}dWGCXN z#iB_R7*Y|6ZU(=OBbOY*#L7enMXN~c5fYoTKwbQ#+6s|Pc?d9Rq;^V4L ziS%(LMxKlbk>y7Z9Xs&}0I7uC->X2nL0v>$q$j8qB>?67VeEc_IslQDy({@Ou@V8T zJI;`I5z7vfQc{EFC!1KgNeeiag7Cv5YTnBx>Pb}u%5;7qhDnUmg0@LDeLpq;s=fk3 zJTww#IMF6Vv-;N@4niX0Oo@@f52~q?>%154^9q?1jt`oj6oE0>LXu)&Fw3w zzH4FZm83`Oiq^2X7Ndi-(I>0Y$Cdd~+@*r=ru{qxga!Sh`XVbNF@FXoQ;w$PN*#|# z-t*l3#i6g2i_yRy7qY3!U2aLVfc(r?O{p~msrdA#m z2+-6`0O7KY$jyHi#ST#FT-OFG)7VdvEqYLJIKOnv+G8TlRUy?rSzayQ_y=9sNAaVd zug!i_ysDABk1tlXnGR%JTdA$1KfBD!H1ZL^GSdIc1SiA$UT!n?+Pl2gAYBA6M1wN1Gz0fxw#pF*(BeW_QeX$r}F>|(FSysy6Lc zE$?vO0`(1omF=2brJGT8{+3`mNw8uJW1v3LR4j_?8|lD-hOpkA6qec9vM*A0$JR@$ z5RM_!Z0)o~bpX2nEj!j_6c9DxGvp~!_rDp1ys%#LvxM@3h_X7ebdRh6Y1xFJpOhhM zvwoO!&$@u`?XGK!E}@&2r^%KrqCsU<#jF=NAwz>{bdFvk(niLq(<@dSxG84TU)8M$ zp#BlX;dPOSVUb;q79>`XD>Br;ZT-5RYlECV-1~5L_U_`)_LB}v)b{vKQ^#~{h7$_p z84=k?XE9SYN#V<*tS(6`OO%)zH)(Zfz8GM{G9AmA;$as4Ixf7Tk?^SG12yoh3XHzr zSRpq<^?V+l=W~B(tQEjlmCPIC!XuN&rBK~ixD|B5LPIk6*1z(S4<5yNfXvvQ2ebLzc0WJ7v@uo=Xc@m!(`58eO0(d;r@;N>WqK85(kT*Teh_Mkl|=9k zz2c|gFhd62)tvyj@>h20Q%VobMT}iHVVmG9W@)ingf$%p=vtId7)5&i#(Y4FH5uU3 zT(9$a=o}cC^%V35#g3`j`aJieNbfiv)ERF1=9Q>eyb$R6ztcc+OoK5WONA(?yavF6 zJUOWt8CyTDJcH3iclHsu=c+O%sg;(}8pYL!R(K>Tk-uNhG$Y-0ab+5`(?tytrOVH^ zi8_kfl_z05KdRKQv)p{h2gMU`0YrPF_p><~8u`@c7!2e%Me0 zO?Gz=3EE^`erX5-*K7lhdsFXxa0&)}P_UCO^p7;fQgAu@ z4UdV(3R%YD*1caIHeQ~4d$(qjzL?}7j@wo#5Tun>G^b|J19cZD=P+xWvVge5tfWt$TEmB}yR}RCB1+GT_J_33I)wI5JHI@(V4I zch`MPlBy!yV%!=Sthp+WT@viKUeV`0VfT*|{jK;6 zIQezD9vL}ga-YK=g0n01%_~CD^4A}rP$-qp#o&`WWkdvr6gYB8Gpp1plVL%d27_8R zW$nAH<2srMb3ToPsihgoJSKTeKdBBFoLM?mFnZYMIHD5Arsn#@Pob7# zcw@zd6R|Hu=H;@?ne5G&E887tR9l&x6ty0bH*HaowSP?8dthMl-*B@O9&HjHI<8uI zg%YvS`PxEilLS!nLh0*z=40hhPLr-y-k(%)pNlRQ8l zLNxt!naHzs?{nyxFAqgbRAHU1H9o^%4J%tNT}BK|G~G%M;xI^K&M83$A@4PuM#0qI zf{{Qb*1c`2IhxW*v@+=;`2_`g`Cdkp{O_rrpSpEeQf*o4xrgG7A{+w#*kw~r?;h}? zQfs77$MWN;H9P5%78bbRerp(ZC+>c}A2{aR!jm@2?yKRj3@H9sLv7-X;_3js3e$12 zkisSm5kUj0#P=~_@dcLG(@|!yzPrUhK~o=jorcTcolWumQ{d|b{frEiR|2&8T9LgN zl^{LQnT=ojH6=e>Wu8)jI35 zqcE{y4-s~+&a+bB^7Z>7X*wP)uLhR;Vl!zz9NnmOF$oPe{+Kg)jQaT@`Y=>VA^v(Th}8@X^vF*W31f;|NS}7_qrd?RmkAm&9#pTUBw&IHaP-B8N7gOH?Pp6 zPE|8?OU!#LdUR4#PZdJyx!~S97RceMtodeGc1IzlOG!Gzm+^O)_-)J$^Aq5((h{$d-0gctG&+47^wb$> zK@OHlpTL>=g4K&+xPCsGT`Og zc<5|}&3-CJ6U7i$4k{~^Mv^upRDXw`FpwBV=bn56_@&$3oJf$LPFSN>Mx=M@Skbc^uW zV67%z@v5!qZgto{u2+0|M0wb18GyE24GOfXbo=p}44dgVabNqkUOn2~ownS2>O8mD z?0mPmIBgj|t%1MZ+1@KMn>~E6XT~`pLs{ebR{zw-D%_dCC!(wkW_pn-dkv42z?TZ`VfKgsATyG+?& zArs6sGhe5y$uZyIG$$|)#0}V4bf*?)I&!;0RaI4zFgnCiN+|Lng6xcn@&WlOk!|~S%%vrsS9}Hj;zd*q}xv={CpP= zm)!4W&0YB65x((wl-GH)1k#azvWyRfbcSu;siTBN#x_29@1WlN&K{@AOB?IUQ#B%_ z=R?&ka*LIE>a_3Gu^;s}-m$RNT~qfX=rFMKe0CoGlV`U0UD@>IB)fY}?J?#QmNgpQ zv37&+QH>Bg#hy<@91BFP=SADi*ry*O^s~pR3P7MSNr^>^tbw45gL8E#U8^Gl1sb_=?SlUHjrlocpANdl_1*tw z@oGIEC@Yx>C@X8EBINgYnbZGx-YXI9+{TE4no$GfRUcSSNfbpA>E;|SOQ@~F{#jZ3 z>A;ByvsQ&jrGDnlWBx5>MSK7s9s2%5Fqgs%*GE3h=cD5?3wU8V8{)TwlMGwiAE6?f zYUB*Ww-_ZlwQ5JN^oXPh1B_PpXq?$D!mCf*TYiO$l67kIU?C5PLuK{SFncyWwtx6B z3opUz*WAm5K$chjIuthKopk7GokPpxnOkNxI&^-9Ma~UyrA4ZSjGttCqsOz-+&P{v z?dQEr7`)6E)ppCivZ+cc_1K>uLv2reb#`5fV}%CkBSZ0YzB$<3YP>s<2{f)D4PcGY zdt+4|1cX=yFY%xjSUzJr6u|4Y+;@)?I$%bBjI`o zBB5ufza-!96JPX*?H{U_aPG#FL2rByL^^ICi^N|UE~&XW@2Q0Ng>xpk=M;~74dZ5^ z1W6)D@hYVB>W<3V4y{Cl>Q`xu0I^XK9PU=vF47c2Su(_WydI3X;z{5q|9irC_um^v zstgL*U?Lips6ndd-oI0!sMJ6e@O%T)%?)n0i^7)3Q%YzBUI0m@;})(O=`@X4X|i>mhLfp>Mq zjqt7OK47tb96O0=wyg%^>c3gz0W_(QBH!WhOa^7m5I|YUO7(0yR+H^2b6Ckdt1~~d z(VDI9J#VCzDs4>a{MzUJiQ6jp+pW46Y6G`ie7r8@O4w*|I@!5CUdp*&{*t_4dhSKe zb?*QXHbE%+`P?}VkEY>H!?v|XVHjCjl>hcaGY#cf3=f|DwWUij=5U^fPajESHucGY zVFUsyS5X0n$a1r1BrlmRtdUclVBfr5TExv0^8BhlVNTf?Bc50q$Wo6i8c9T$T!K$A zbpKMZ6v`MV&iWGwYPsv@=JIK7{BDe5HOf~)#3?<|JI9uf~Mg;cg47EwH;DE`SXj9knpEZw~wd> zkEbu!yk)Rl98@hmjmRx-x19H*mOP<|{45h?jza-$iTMR&xT`tATes7PK3Z=nLWqbB zRe40Il>>qn89V8%UmghjSJb;mI;hULgqODOyHeaNqK|j%hI8yOZdRIoXKPM_jaMA1 z5RwtZK=DC@>V~si7wbAmS3C+$y%p5G1e4A0`z{kQ&CWC0J@JUmL$>)P(W-;8cViLi zt?_;1xc7DE_8pGA9Uy`$Ue7JII;X3*6o{jcZygCO^nm-)d*N&Ehp$}-etyAEjw?Is zEep{roN-?#99rjLD}P#f-6$Tu$1WnKe>y%$+0Z~p`>u#^ptA<62OM+X!&_jre09-~ zM2mqD4`)CFTrz>2pDcE9lsv1RX*EXmoBdEx#Mk4ViYhd*@qxL3gfM(*U2~KS@=^x( zOdZd!KI-rD#-m=B?aR^{e`fe13ydtsC$;+yrIQWZ6h%NQX5bK4eTaH?jJ=y_*;rCt zH4nm+onib|2S5&e7ycpS2#QLz-r;?@%l2X)|Hm*Mw*SRs^=+Qt*^ZKsQNQS5TXD@X zoq~agU^<$F=&w(xSQ0eYNSS0KRyKD0@zo2>Y`Zp4v3{wmjp`4C{`URH#*)2nfwC;J!+9Njt%eBe#b;hZx+I7`2 z;$_1ES*B6!D7FfPH!7TUZTtchyK%V!5BFK&adEI{Z6MNij6BXHas?<*o2zfC*m5pU zCIFy}9q1gMxieQm#W`0qr>EJA&@RFko-KY{6g=XQoHl5j9hXS=QSb`%aCp=0@93)O zT?>I6iKok7C(>0w4VcYmF?WGZtf9Mm91@q#dR8sAuQb$l*G>OyeWrSv_rH&eZlJ(* zMotbPBw^x=U6&h9q&R}MF2s8UnWCS@&O#H&%`Oj4UTba=wxAHi+YMP$KJHQ5#A>WW z!p5q%*qxl*Z0=-MycC=U4WrG5@05O-v@GPdJjbx4dYe2y=yMDqfL`@o@dfcK) z1*eo&$R$E*8d+Q>@9@-tE>nY=uME>yhM=eWG zt>srLDufFk%d@j{{9fQN{}7Z-*jkxk5%fTz!19%{id!t$`cuSTRF zHOvs*%DRRQE$IfYCG?qIYWvX-ZDl?$?!E6MkWK+~ol78BYBI=HuaTF$4w|6MNrfry zjr}pbuX0}U<7wxmq(CF6eIFpH)r;ZGW1^ZsfMy5S1>ZU&H;>sPOeqqP_0-|eP|qpK zvbkOSe)5hqF-T*g4yj`qrJE4GJ>TJijnt*(Lxz^<v<2`##~T)oXv8rWmY;QZ;H^8 ziwz7EhlTLPa0+D?i$|L=igC-qol^OfOmnuT0libqB-AGe2b6zDBHjS<$q#M`0(chi zRJJucO~Xob`$$8UGwY2(%g z%E9@)`5)`BFSOt_LK#)bbLHZp;{Gj5>7qby3u8+2`Tk)G*rejo5qjtc<{QW++Pk?5 zZ52#T_EfC;F06p(_nK{A;F7&^)by$ZG!F6;?JWEIZX8d~RFAoAX}iuw5!g;z>h3o< ze&@kd3|5e+QpJ3i8Y7Wz31kLEM=iEy{`YeNYV+F1zEH&c#Xl4fh;sbGx`majgz5NA z*UJ)#@cq@x!^@qpS$H@GVm(t#V;96n@2w00o&BitFIW~F86~3UJk>0_!#g~w2>Q6? z^pW2eF$tMp|6Z(eG-Ke_uu0tifJ@-SP++6^s@#Ce+dFAS5aHD!<`hRIxEU~q=vQMt z@*Gq7O3VzMpou9pnoKPP^CyOwVa<1y#A_QAE=;WeJRe~MuMQ{lUUDtve+@bcZ5+U3 zyM+DsCnMF;L2>M;%3wbpgM|kxv_LKLKs^;R##hsl7i@isYnr~SGZ*+|G&9obGf_y; z-#p-yy)tT#5YRwaPP6$*{HaCK<>WKy>3({o$O69fc)awO!QtJ9lupfuyDfp;6F`6( z-}%W%orXW6UFZ!pCVo7KB?0mQEGUfhxK{LJp+v=I-Z;X@Jrs4YzTW$A<`0ZdOf)iy zQOK?*mRP3w(UEHC2S#z~2iQsEk#D=eN%nQMh?~g16YUs)CBfM9Zf4a8jWc zzAyeyxGx3QM)9Rqsp6eX=^sN3W2{OQ!}Vn8Amam3AOzKYE3YQu+w(R4*}I~$L03ON z+prU6(0H%{K3CKq**C)$tAGB57v{SkuoySd#qa`v#1m8r^vsV{iq_F>`7LYWXTHy@2G{2D^~` zQTEgd&oN5sfIo)~NGeBQ&2ye&T0wTjJz~lw@RN#4*qf$6aaCG^AZ@Mt=J)fn5!gHH z=TJ;DqTRvh$|w@~YTF1;E#lM=Y09}6F$6L$r6w7&qy{FEI1}_Z52kuPKYQM*Z5;U4 z(DdF%EA`uJQ92-L2NCu`T0|)`_YWaaNinnRH>P_}?q@=65BQY%)wlA#&okOgM5qP@&7%mao5>x$O7g%#C;qZ)2nHPg>HV>X zn(@mtT#{bdO$t31KXbVbL!FPY;#NPXsCCh+T+E4t%)28@22&=mqh@&C()%R zclBYRbwLf7(lrOh`0C3SRCHx@8e=;#W)+-It5kI#mKfLaQoHYuGSDjCW-W73#pp6w zL*J6#Fo{1>AbDBrVTxu1-I%5F8_<8Nbk=8){!MX8b)w|=^lNca%Uu6`^o7Tti{U&| zEg?O5@(XMmyQGK6AH)a;3?*tQ%-R&oJ~_aUd=30ZDLg@$S&%grepUnIt+%)E6LjMx zy!+*0$tU5X+(7e<%u!V{^ZYkESL?9{f(XgrasnhgBHW4`ryV0NSN2J&95(U8n7v@5 zZG=ftq@J()xXblnm{U$T4*4ww9>C#03j+LXH+ENP4RljnIjL&d4IKmalPZ^_b`<|O(FL0)y^^5 zZyD0`?I~;sRguy}5bGE#58NTr{p|3lkiEAg-KS8cMpq644<4CBkF#02r-L1(b?>t$ zgQrIeFgOm*l`Ln@BSAw$=>CrS{x-Dw)(~yLHXUG8PF3)tG)k=CSkld0jbPbmZ2(|*u@Z0&7+LwP_-}BX;Xa!n)7THa4$$f zQync@Z#n_@RJkK#51>l!N5ArosvcP{X>5{fS1$1PynoHOHZ$`l zJKCw~@~jXDB)i%4r)E-JAlN2`_AVGEe0zVfRNLj{CE%-@ynpqWE`LOi6ciyF+W%+uUDY;*&qyXl2x{Dx4%>IX9$qM6JRVtp z-kMecFTR(F;ZTtIOk>Ui2qhX-BBw($&fJhr#(M{{`8*?K(N9VnSQ|sjy_tI>#1Wiv zBl?aimWvS|_w&w;@a?%q_iZR{rn#pcONs=VX*>2Fkeyq=oz&R4ql+*_-p6|0!(>o8V__;2riB zH}9KFoFt$Q=g`iNk5%PW$kti3p(o`+w z0ha8dqznMK>VpttZLgybSP)0sZXbb)`VWbO1p}p%}Q)$$Mefgx7gw?c!Hu zILz;9V#lnR{VV84c#_j=wd7>140$7Q5U`5(2ob$Fw;)w?Jf*XIbS4I6dJuNlhDmy~ zbG;MBi+g4sNxdLD@hiegW$}9>uSffBl*;3;=(CrTvllNuQj6e`CD+u_yR)Zfs_$GC zQS?sYpTACiq_O7WNSCi9=Nu%o_+pHYJpfXUCzBl~6q8O2K_A-{lL+e<)glIize$N3 zEY90<$f_c1j1lCwV*noyhJ~N*D{G%G&YoMR*_Hl0Ufy0jd#EiEHBKwMnd_pIR3IjL zR~?9$qM?i!ght5uWBfbL+3HBQ@3<1lm&zS37)dD22$+Qt551nCqCAT6N0nz_(@?2f zsNLsgVIHAjZBv0WpC`l(=?4dE-__>Ft(HyJRzd%+k0?!cO?_wu2>C?nMnN7JJ=xbrWzN%_pg+11zI`$>a(C$?}cr{fzICv)39XkrkE zC9uhyNY1c7~R+6&y&T*P&D2y5m_ZC5i&(hn=UHLWCO{cvGyQ=!{^M;);IhgPkb(R zE7&aV9Q~co%B#Yn)aGBIDuaT!#Li3!Q4Em}Avz%d;`N`;JpK_;utk;c8zujXh}Y)3 z-Ylbl1J46ZLu5s8PUt<$2e}GMzg(Jf=y!x-CJ`4mse)oWy@Yad6jyky1c};#Mk3(! z^^_pe{XI;DKyqF!+k@N05^*;dAvPCb?L+>9$IlP>-YBlHfZzZ8#jtls3~$XvSSx2G ztqcoYBHQ_?N!3(kJ+IbVY0ysdepQ@Ehu0j1lnKH_kon1DKx~NY+#~(%5@-TNNvRAs zk1{Q8GIc+$lO)W2DV^KMiROr&{cR7jcuq5ZhuXp!&86$SlA3|YfDz9(7r6;FG@57% zf<-s8>==~b&%#q_x7(%2{O!^((P*VDxd2c){2|qqvUM{h_YYCyLW*t|3c8$G?b}i|1=J;8h zNca1|@TC`er=hf`WWJwHS$0QFy@(Q`@c`Cmv8&=9) z3}HDTKww8{S3zoTOKRdszZ&I&x;cilt0sBw0Aj1?yF| zTnq=bhs{!5%4VboYomh3h5=ak@w{J2=9w%&(9G z_2QkLVV_aNg%HYnt8Cn4S0Pf>t4fQSGBPc!sX0Ggs)D-%1tBu%9A!DAtZ2nXw*%JJoh zvmnx~=_1kVME`2t{@PgKQ0cl^3`v5tialELqo9G>j;M6cYH|xLVj!t9QEXU#h1`35 zy~b?)3VGS%l){45kcLaMPkgRl z2i{lr6C?+$X2O18s3PO>biC|}Fy+!_r{HHUlj(F%=i@ASD9ghvW|4!~#6=%vA zv*L$t4)08_b+{+dRUFNHbQks4KU8<;{t7FAF`Ku`774xy(m?*;joUag>34A&Rmk`0 z)BQ;!%J*W;#ls}VjpvKpO- z87BMHZ9HE=0$P44dF+kfzO5#{vQuVHD>S_P?cbdQvoJ#0u}G5;haU9^c4w8cR>1_C zNF6B1DR?>{>8V7?D6Cn7hLeyqchgEQM7f?xO!`f3y5yJ3R{F@`Jb~>h;w@>ClZ`v^OuO zQfqqm+O=m>%Cm+e_z*;{Gbz?PZ~HMTFHb49s)U?y)F*IBs4Skm7wxy^)g@1>#XN;^ zkWo3hXx7@-OulzE7iMUpv2v(ItL?!BdOMZ}vVv3jvOG6X}{-Duq^w2z*QS=#LEG?2DonI`3n4xGWCxHSN5E&Dt?_>$a1xQDV%vrLie#x(CwZkov1{u@f1!ky8gY%gPu-*j*c^NDO8aAA~ zz}EAGWmO^Kq~HvfLNb zstNPGo9UnF3axZe+S&~X5((CYGGcV!jZx7$Q9GgQm4dCBzqkD+A2)evXd7uTh(cpV ze=xmNu`qLT&dk9;&ve_9~^4M{(WaCoj zpYi$MqfsJ_QvGyqw{b<3*%SGdTgMoab2!2n zjf^gUoZ|vCnQp5vnF2%XuJ_Te^QaC3M1xbU9_eTVBk%L(9!b#;q(y|K5Vcc)zk($N z=Uu@039(F7*y7xQk?Cq9h)nCa;YevN-SQ(|U~>}?GF36>FE7h4;-5aJ&PPP#apsHa zQ+Y)&;LRz^cM;(K|8LPjN!D9^n+eq!8sZ!s!pF+|8Vgg)1_hamL;7=7mXJ1$m|oW} zsWC+St#jZw5SJ!`QFM$(10=(#u6jX7Pf@R~?wd+qJ@J)xi)Aa${$zD+V{PM*Ujw~$ zz9}?|vf?vhgQ4+vr>||9-zb@Th-eYLs+`T{f$X@|mbr24$W7C-?(Pn5nTd`nSl7jC zJYnI%oHDi=W;uy#*qQE+`4OU*4!dqtJ|e?_hCn|ZUYXddfA>pXAP zuk9Ti9QGMXgk*F?_MlhBB4H7Mn~6-^2-VbrB$(rBh+gWdvu#8BcW1NCvHN2@RD*XM z+vqxZXaYZ8QHV-mSkl&Q%6D3*S6r`HS-F+3zv_iB6_a6wrvS_#=%QLxp)fO90x=QU z>;e3Mh%$V-fB;h*T(bX1#apn&0VvCYxVyW%1b2r)hTt%G(BK*fuE909Yj6hl5L^NT zcXykh0~uWRJ9qcKJ+J;kbyatNEmyIhV-sWk7iS_|Ow|U)v8Wm48;IDuUJW9fTePN< z9qZ0z=kZ`Jmy6I)CLpJ|RNUEb-683jkEsUT%AYv_V65*m}C$sFi&f!id@f zcfwMGoLT@C_qeVEHL(q%&Gvm~d625J(UFMTw(s%05v1ngBB1E~1@gh{m!ZC5*ecwB}q! zXF?KWDc0BKeY<~*TC{nFecAtmqP>{BDl$$M-PnalA(>WDzDFCLZEOmk0QP+*X&1dq z@8Gd{O8HB0g74rmD^=>x?W~AdUtgXpw)0IT(d^le!oo6+8aV)JvvBbNrj$%)BaBpnk4 zUr=O>G#Ks4VW3-i3=}B~7Qdrmp)h&^%m26vCBU!27$#N#rOY>GOzx_Q!0Q)ux!W~t zo@61iF5A{!Vw7`NE@Re82{b!efd8=QDAsYIIL>jh>&ZIS6~vcTu8m(BMP&KsW$S^# z=J+YbPV-#saotO@F|Q>l?m?wf!@cotnfgE2Om$E%8mpR&@_A#;>fWs*hYmz$oJN z$@?>6+~VWY-)?N=VAN&YIr*q(fjnFnrGrPwknqkehk>WW(< z&wRkOBrJiOj_!q?rE`M7E|D>pkx*O)dyEZY$6`})ktFQX3)llTMuK_;{@Gm8j*X z;V{8?Fh^Tx1~9|W=oCqfzk>Zlmd+(6(HGwwFTi@0>DM~{%%?5qHUVKDx9y92Z)pk~ zAIb|*o>zvkDto^cq9LdhFEo0{baWe|ER-!>HuG3*UF_i1(oF3vsjx~_&BjlZ>}Dw= zijaY0qJ%BJuEoT}uA`ajmT}zxL^^{_<*`lHC(LpVAG(I;kjey1Jyg^{U1laUXlPt! zSx&cEI=dAX7HN>Wf4T2{H?TRNa5uNqOq)>zeu6j?iG%E!v{_alD&6ciX6K z{B*$7&qn&`M{H+w)T`cXa8)sU?Al9bxYI>4?}PC?1x7oo+glEGZ7x=zr2yJ`w(e|2 z*@ZxwSwI^}S0+1Q?hqjS`5^5VY?zCY~Xa$l&Q;{35Y>no#50YD(y~Z*9J^s$FQqnvt8IajliZD5P@7# z8V#1Rgp5y?6?|$sbnNuP(6{)-=a%SRK>I^!*e^p8mkHZx_rW*FyJi`8g#%j zuK#O(Qr;;h0;s~juqO{KNwtVyImR?^{ei6 z7SXMA*{fC4HDvR{W=CC0rm)ATI;j8z%N@X8e4IK*U^aP4`!8hI5l2<--{Hl}3?l9-5CM z;*C+(WBC8D|AeiH20xWADC_(qM{c40MJuXHB79(WypfBN;aiwOD{#1k$*e3?`Qky1 zk%G^VHks|4(s>)lSS*DNk5+P1X5dkQ9f85|(JT{SOaI?_MoXRk!H}kFkO%8b^m95{ zD1RCm3LmCP!=^?ahCT*68VE>i!3O$HYYb0~_P6@A$+E&%q{Y#8te7oo$dcAt06yrd(2O|~MGlNWjjCFocdzm`+ER5BrpJrZy#_(&@3 zif~f@WuT^R{f9EybCaw=p zhE@9(e@2AoJKLYp1R*I3#?23l%-!e{L^B`#WWTPB)EtrHX1N4bDBk->9igVuCY5o2 z@6bH;52aCl{3!S)nfa)qdG@-bY}_{`5L0*u!OTs2os6fCi5TmrbduENp&hULwx$}V zDw+KUj}twH=#;iPGSela8E{K)1)8A~;~K@b9hjtAg*1g4cgIxSBOtdBqoed~y7+LU z?7h_k^C%0JTs>=Yw1zI0!6wX)Z5^H?%PjA2c; zn1e`0ssiF!{yE@Tu1^{6?4W88{0DCOj|A?&dV71lZKqiD<>c&VRLgG2+%jl@IP73R zn^Zf!t#xA4be6S@>)&tYo|!^%C2*@`5`Q6f(}FzaznYufgF zY2zxG4f3ZzPPiyOR2c#2gJi>yB>R*Ld~5%s_5j?>o}X06YgV1K=+>FZpodhSa_=Rp zb5)F}Ect+;l2adj$uJoY0&&Xxv+B@jc3nv-cyqh+q765T{-gvWzXcA*y?7ViC3JUX z@9OF~U(l`Ql#v^+^7Wh1cRdE!j=5CE;mm-q3TBtNCQ_QPNM9MIQ%DdO7bnTG`FEbe z%9Uuku-|nawA?g*MMr8_TZgzF{2{eT(mEme_x`lf(phz4q|`V`3_twjM>kkKq#FPF zE+dZ)X%^v~{ZtIX{Qf@b5kfkSOIZk;ZQ+^=tx0MOag=2P!1cNPwonnK{3)fX{}g#x zzP>mp0h&1^{+xZR%3l%J$;JK^4o)k3*dQ5)5br_)41TJjNK@ZBZf#zajbW(_Ch#qw=lfXHQF zqYUroWx*4+Tv?21S@d*-9t`VU4kEE;BDi#?IHPa=n98#JR+2-oE_gUE7*c+AMA9fC2FXEt2*}w{axp$xNQ09Znc2H#;x;f{fCh>16>0B`Htv0G4JlZquZ{vulG98~|rV-l2^+$Ua- zyuA3u3_e9sy0574_VLe^SVZHG^3twiBn)ySbS);?&OWz~N*|VFb83HLGy*fp9W%-m zQwIq6syWfrKEo07vf<5!=0X3>li!G0uIO4CrHk51PN7>-^3-WAIiwTsdt-DgOJ_XKnuiNamL1BGtI+yDaYc z;*lT|4^4{i*;d@y`F+3RSG3bMP2V92yWKpa)perTBV^3nZ3R7&Y-9r0SsYAi5~!p} zuaQD8@^O=&(Bgjhr9N0j{ z&O=}=0nRrpt*7%1Lqry9#czfqC-(Q9sqVaRJl;&19hG@ zk1|Bph;A1)-}Z+c7@>O-G{>8k@$|}kxbr6s3L)JI*|W3vt|TaCT-zggQU{%1f zG5(inKS+!b^tl@xbw}QXwct@;U`$%^L8Bn7Gq2aRw=;=X~Fi%){IE_I78TmlGiz zF!j*JwN^~c6?HJ1xAH=0N;N?2w5*giiOV$q%)2Y=Lw--N6Vw%YtYJn`XKd6wZ_3?s zE04qi@|`IgF|72aF-VMTaBEV79qga=eZ3XiVtuil>1#PAWij4`1st9QTLlon*?^YU zDOD|r$|f^-V)Io?3()^qh@KB*Wnz8H0O?89SjNN*a>|j*_()bO7$`WFAq zS7spw1xZF`hQYVlZsm46{3#gWxc@J-qv+aqro_O9mGGSCC`pe=rv2C$ zdn#A|bcd?0kwd;f?ul1h#-6}EwVEwQFK_8zO#wb39X?L8W_|J{ETs_?gNIU*Xak

WcEHEdj0g&K}X0!I$&pi6fGrwG0UH$B5 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

    adG6&`4jwr&ymRcxfB6SX(~FB& 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

    LF55y`SFou5mW+mLra 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

    b+8cZ#%gz8-~l9=oGINY~#!SE~n-X9x~Z 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 z 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%0A?3hgcKoh1#47lM}DI5 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
    |Tv^SsQMc31va-lWDN-C6Tb1ojZ#2sL{eHLG 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∓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�>UJY1W-zwM@cjo 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%0Ar$2 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Ž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

    O%FnK}rEb#1iPu~sS!CU?8tu^XXY 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

    cl(=R>$@++&o?nJw7l6bY-X(UOSrrplk?!CKz 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

    UL5rvP{(< z^=`8-OFiJPHWZa`fWKjgc<|2b7za`#Cvew4+eu)yFHmqScH^v-fIf|V5-yY#!2G5U!*gB@)o1m|CvkctNJ{%nQQyBItG(VRN2`o2F@{$r6eDM-vc#w`N#G z5N52oi(zBvxdb6X6y|X87DeH`?{qreS`{}KT=)zOaoA&xP&uHBJS&SF00x78tJP9* zbPrMhKRw|Z5xw_iSvu!xmh|u9ps_2;LcG&N1RN((+D;&epv)l~i!0H4@5(HjAVCY` z-v`kd7Wo^_x$1-V9Jr4S19~1^SmBfq_i{i>O+%D>hP8qew&?zlLZDldcpNwnyW_fsRuh6QKw=&)x@jRR~WL0wC}`kB2&$JLd># zmL&fE^k3QU(X;136!-n)e>duLRarHo#0mey8p|>QQ|OCBWYB;fK@@@kI5;@^#YZ3e z)|U>SJ&r9Jmf3K$4cDeue)!URKYkJ3zYKAwmv*y_;l$(e>Z+YiN|vz4R19bt3&$+V z!~%G42^EEfJ1@o+5xipA)6YWt`QdR-=w=8ARD0vvnudsOT01eD^M(Kjthamq4@fB> zkp`ah!y*$KZt3MtA#NAkbQ? zb~5O9R_ao_PzBitcAs^>yWzwjVT3tBjK$EmmN)xoDI1AoGQ9_&& zDpVEtLI^<-C?ccDXcb3sD^1(2C{3a`j0Mph_zP zwA$^XmvBhKD>&RoqtT2R?Y*_Oa88_`Os1}Md7e+|SDxq2x$$@$)~`iTlx4}*0yB69 z&vOG9Y6|hBDE3AP?&`djx#?FhQ`I5CYQIaSCcRX9A*!}{O|NC`y$2v+Wl#vGlQG1Y zKKJyq|LD8@hmYNHj+}r0SFgPE!=J*PZHziy6;IrREu#|wkj8O97?|Zy+Q!bxH8dPh zwr;Ki;hh>iKU2mliOv{f;v|mZI7t#?%&9XEM^TivTHS89+wC4$Ig%uaQtIhA)>@l5 zQd%=}Q5IR2mDb+5d$(yVAB{#^TU*2J?QA;z_@hhU86C5AVx2heofl?>S{fXx2nx&? z&_PWLbWV}VA5r%jHKXXdVGbKNeshkeiOV(5xsSyHP=;E;l1$R2jF1tV97%4f3BHi(IAS8orF+1 za2LA4WX27+7f2yt5(fZ8cBRY+tm*_N#v`Lb<)oS@(JUWm>XP2*N*tWLoW7TUt{p^W$Q-6;PRpL-_ThrPWtZ<+7Kt5r&sDPr zOC5ptAp(i)et^^oIoef$7;Z`&?$*`6HW55w3fkwq1`TuR3E{wbk-CSO{e1TzAXP9D z@u5``YiT|-6FR8^ha?Cb$Q4-qoP_8J2+^saL|an6-EA48Db9DM+LxeUYjfl6cP}3K zqbI%{M;2eEGSre)ar?NAjYaOegJD8pTnZ#4IczaT2?U z3;03LUK`zNx3t#QTB*;(PxepE|5v5d)hk!0)9jH)9$i{pE%G8xTi$ylCG~>7Dax$I zVJ!Dnt~EGxTl&WIDVXSagzM5Fwl9P>qwa-0`}r8 zvn%rnkG2Rr0vk>`8L%36mATAR7Lb-C98JzvJY*kmZO z7v2Q1NzN7l}rJ9Fyf194=8op8ugR*!3DshR1Bq@p;D;*^Ifcsd3E z6-0kUv55d+I-Zn8X$uP!DPt^)^B%Q;h#(~J; z!nHN{N2y#PNFoFkDKUg!1waVyYY0sD`;p1-_5|3&0UeTG;XdwzFDh(EgMx+&>x9oe z|JW?NWnm#+C8B0oz~*OC0Ek{fONodEhyhY~^3VR@iSK-+e>}OiK2c*>?zeBe{r=DY z)nCBtmo=o6E|;U})Z13t0fF_6Ei{^)10vEy0%92fV-wkWvEP_Y8nebMAz7Owyf7%- zZ>YNWJNA38mb|x>L$ir#hVHXCnISJLh%eVbpGAf;q9By6oG&F?m_t@fIsuVTf(9a( zq`RIL;c-$-rgbwR0ugBwawxtmn;3wayQO2*w=}bOhXqMlR&3mmawGvG00O=&n9x^E zvlLkAOfUszVmNy6@$Y=?3xE0ttEUf)Eo`q3|NZ~xFGk=0DeRQ#0}u1YNJd4exXelx z4^8{#+) z+O$$?H86ssDA;(DB#E>^9Sx43009+f54wWtfSr%DdWO-rB+*uFaDuJ}!dhp02Uw-ECXTP{~ zxVJSbU%hzwl^^`<_J@~?jp3o=53i?o^Nm~)Bl(Zp5 zMx$1wHARMu7DaI@b;6c?>0RlpbIy4Wp7WhCuz2>adiiNF^~`yZTWj+?AC1T3@pv+s zWLYNGXY1SKWv;WZ+JQh>E*PaNw>W^~{Vw1+x^Znw7bMcE+V(QDZ33R8)U3;>`OpDW znlOmWYAUO%SiFb2qY7)bI3fuU2x8NmSrb*YF^wnywJ?py)5$nqYJ<{neC^S1KKH<* z=bQm3q}U$6^gsMHynYdS2WcnY+1Q9$?N~=_EdVf}tBaIs1TO)g64Xdk%Pay=78Xzf zgKPewMikOs-a|ftos4LrwJ5rL@m(McG2w&oMTL6?+agy}={iWsQ<>lper@eOg zkRT|fdhO2I%7OI!g(OKLW7=ur9sAOjSzcy&k>$25z4w#pv?z+5ot@3i&CQLCt&Pp` zXq0Dpk>?oXlWJXDErr4ssp-;eex*9GHXni_0mW6<9quzp-Yt*rQ^iv&;CncN5kNC2 z2F;a(#kHVb#pwv36|I~)_Skceys6Rq~vM=pTYKLXbm0rNh{pQeDHDG336rjWHkoM_!ya!rPrLl33I%h@@13}A> zkn~qdX|0s%_gA$sX_~g%?RK}*?RMMkwn>t>-4eo3X%NPNsECLLgF#jG+T0-!2w-O? zV}%*{tjt}WduEJO(Kf9Imv1sm#79Z8b#uG3HGcTPlLsdF$It#j$|~|8B^P-aY?=0WU{;pd~8~pDGW73xHvSf!0g-(txq%Q zSU5po5|07^)_Nf4Q3@ibOy*^KoR?NSQ&GBDdvFdlA)})Ds;n+A)M-54o^)0Q51n}6 z(4hm@uivy~p>+h+NP_FpT{yD*@R<`wj~xO6=d5Y9!53^n=~(Q&fgcP?ZIMsL;}Lro zn*Kz*LgI!oy*M1*?_xv1g^^%52N6gS5^Zj6y!qyvN~zPQPdQr_MQ)4<#e=~Opp3etX_}75V`ps~#k6>-%VI>}_f1>g1K(4xL!rqLDg{a-snFaWOsV1C z<(!*jQzVMh7^yOAmA2F-R`fgj0iAPMo)<;#Y$+l|o`Fd0T-0vQt#N2q(e567uo8A~ z3I)ZBx4vQsitKe-$=vvC!NrLSxAw}~04R#UDCM2arW1{jv=SAOo3a8zP2g=Y+F`ay z5-CN(^Y#S6Jt-VG|L7wSU9YrRfa|ITsbK!qGYA(IoxMr}_HH%7P%#)P?dz#b)jQiT z@E7iI3(4l04W_xU8Y}~d5LH+?sr5B4Tzcz4ce|Ziw{G?a?KqCsH#W6yx7w|dpE~aW z38eDh19|{Plq{PrFR$3bX8Ck=b@jr93$MTS*7Lvk<)8hd?@Bcq<~j$-l^`x&Yn3Ju zi}yAk^xBU;gkJ-i+Sq3 zV|Ii_s4im`YfQpG45Un?OaeO2vtnmrfArHAR@NT+_8T` zje@qYt`uOE#73J~#c@%VNEjKjvAJnTxw6cslUQq|lrX!p)QF@>oF9&co>>4y(8C_Loe7vAN*9A(q>o#B;RH(Zgs$#i^obL-Zf zojV&@mI;wWDm`-Gv9CRQ{PFWX);cjKE}U$y_HP}1;9~K&a%BsKV*usYyU7qx#htdb zj#YIaA|S&9BxsPldaugV^*a z3Ni%zNoD9J4T7LdKAli(lu@3iN+n1jo{<2nc`t$hDy4|nGu&rR=f3_p0ngniff2c` zO)(}C&n_##OKf6Il(7Y)flF|n(*%?N=SyF>@Qr`=X9pjDc++QYDo0k^8y{}|`~UXe zUHRFIaAyd)a;_u+rD7z^SP+ns#IummVNK&$K#>qQVN|qB38)wJzB$QvMOOFT!S5U2 z@eO5SB1-NNfkedPuG+}jv_w(x za)z-?m3~Db)WPH_%=6q)J)h^Sk3R``q(uvvs^O#6CT?-^{s|Y0-*pm(iGi;ZwAtdH z{&25>hv2?D1fz5yqM#yD3RT7aKCrSxnsgjRNt(7>?X=yB(?pqm+F9!MyHP8_xQzWu z9ARWU@MB+mD%jgh5=X5x2VZEUe%rMppXd@Hw!fsI zohvPSA50TL04DE+d9*VnmK$3`wl>SN@nkX{SB~S-+U@P_Q18Gjp1t=zlsAIkC#AgS z^(YACJ`SDG*Vo1%!vc{>&49^6jaD@%62>xch(mum2nP@D`X3uH&Vv+^Jr7$1_N*9Q)lgkwabK!zAp!Q}?F)lPZc@XOY0ao@3g4rD_A{HPM*~Y7v z>mU~yEm%9M>RKxSSXo&~({wNxtSm1tEiHv&bd^NOQa?5NU~(c!k|>UiF{Bg{ZV!iE z*aiwpo=vkX&vUkRZE ziZyM!+^c~2o9$i@K23x6NrLZf%o7nd+S`_uUp67Pe#*le{t-xY2lso);Z^XaqZ`}DDy0vjwgYh zR2GG|j;$-5&3qXd?RO@5QQMj}M^ zj1`O;5TqJ9A(Xxh(pzOPZ62peg>9+;x^S8hDo`kl9zm*fQYfU;L`^%kkYN&L_RNla zhSrl*e?HvdSr8;K5sK8RWwmBxl$^Y;yA$S@;Nhp;^}J;SM?e6PL&XxlLl9JjIo8-w zdbmA|t+!qmN*qXMWiYvUv!lczr$%L2&anwL)q1ckB*IOr`2uKt7hq#yrBKh7kZY~6 zF=8}j*Yh?A6a*BlG2Brqw{d8`t#6 zgWRucb?{IuTRO6axIB3H2!W148C0m0Xk`s~!cI3)0I@(6Yybq7008Vnkik_0Y#R)K zB3zVZvrqSR!c3TVHizteK&4HyEDwWu=ZdoM%wE`e@4dIqF*6DeZAq~J`-<8Tref4i z4j#UHnqH%tukgo|8CfZdQ}YB#vVCT;zq;+PiA0ki?0zHa1Z;ikX!J z&+Bct;Cw{d+}!--FMs*eQ%^nm=%WD8>vYRJZ?{|3Q`tKa0cM0?_hGE@5%;v|BG>mH zr4&>F87RCXQh`5q^ZNDVx##lfC~37DHg|K#@Uqmo$}y)zmCL7c>Zl4sL6Z6|T;nD@yA{|2==xTk}-Nr-@Ci|qE@ zJ4cQj{{Q~#f4z3#K%{8n_H8279}L#g2#5eE%X2`blm=8wY>N^DHVhTTDQAVzM#Ql%1zc%MQXF?$emYiK z7g+{ENvCV4MNyQUl@-axyL&pT@=OkQ_-e37%3Bw5vpR|dpma`~2mr0~NXmOQk)BLP z(<~48!Lqbb6tQ>VITCi2BEX()>BI{Hsit})EN>YNi+HD_4tPSsyeQj)US=l@Wvkux zuFPE7iX#y$vq@yk#CuSNbi^p*$!KkLxh$v7WpNaZcDB=2OPGeE%?FMjb)LWfgCG3j z#h1SM2Vb{kKAns@t+Xtrick@fHqQHC@E`>53IJrFlP6XW9_pSycjVHg%O8Jybz^2|Wh3la#R5Flwn zAY=h#=bY$BbJg|K;MfEa*t^O;+_c|_h=3wRFpJ}cKs$g4sp_Ov3bhIzN4B=kSyp4( z=5`lF5dZ@Vp#UHYiEv>pqzFXNUjg>{@YcrIt}JVU>FV*EioPBWPg_qxb`|-11`r|)%=nEIvLch~l z{_++?xnM8B^ymVydU)y`w8?hi9(9)GT35Wz}#s4YZDB0;PmB0HUc7tkh&BXDfXQozIJ4jfzp znZWoCJ$ClXfBel;Uw*v%;L4DFzuR(SdHIL0zWFyl+j;Fh7`cJ&@06n&Gy`60co4%} zvs5)$@Q4CJD4&$t!9Aamc%L7pd|K~2?%Nb4zY~|d0N(fC_zWKR3WVqhtOzR541|!4 zCq>4PM$qqrLMIG~hUXr6=&1{jeC^p|hkIMMv-e(k|LTdPGMc z*m$n*MOF&zfp8U^>vZ3- zXTJH>HuaMznNG6d%{!adZ*5+?IlaAg)CEArf03nJt3*jZQP`FK3u+1c6I*%^<=t|;Gs<85*5 zt@F-#=ftxxqliwDkol-qZl0MfcqT!ul`*j<73h$`cR#k40Gnc~Dz>a;!lhyakKRKB zhzhMDr)hh-9UbUTfyeny`~0a3-}n+98LUso$I|YLKl$Z5FT4t(ViR-eTkTe?&DMD2 z86BYrAQ1qfCI-oh3>vtEstUPC;Z8o5`36*~tJCguI-PdAog_(!Zs-vEh~hY^&_<%E zv)U^KxSnDMvKk=_w@2)~x6YQOElO)`?i^dbetq2q!6?h7lW8`c=6UX|yK((`$W5^G zRT#p1@M1uTms)rQU=im*q}S;NAu6b%Qr)j4HYP>{mJ|?K)1-8eLBH4Pw5`@yf|>Hp zr5iZ1iN#2hF$p>u4u`|x@S~3}RfSDRy8%f{2;aJaRxwXwc2+}bXt*?4m(nwc08wXQ1cnT60dDcZ1|4gzJF zISd~lks=h9!k6OMXbsFYh9}ggRbnkF(p5=MC^-nw$B-!`5f0*X7>qe~h(M@A*0^pa z;#Rv=!6E}i3Png9F%miS!om_ZDJ&!)?5(e!=2gI75U}cXr+M@}%DiiAG2bkROxR!& zD(M+#pcn{n%D_1$aReBf62vR?1Rm-D0R?|mjV#B?Y*YshWWU=WE{@0W&UzP;L_nyt28D$MP>c$J5R^8?L~*2|=*hF^ zO&qt|?OwOn?Q}Zrb{xe<>!m@=4r~c!S=yp3ilQtEXZ_8qmv_dKTkCf=?%d6{hHNb= z9JlH2!RVX+`~T_M7Y0*-lHt9}?|koH|J|W7F2-K7jvOjRLG+4pjh4U(YrvGzAP|B_ zF10X1J83DTs!2FW;v_akD@AFmtu$3Lt0+=RDXmap@6b|+mnp5QLWt_(QDL;NwN?hT zh5)79+0KB49fyW1=SwfS$oQ}R>R*k1c>V6i-H$)MzO$1#C+r20obw(mi^KprQc5Wi z@t(zHq%^Y=D32XG{N3;VLA%}h^^32LCsSK`0Ut#=ZN(=JA9>=5$1Xg29>C4TNb7Z! z5GfTn%>mM?lnGTOu{C&$iNJXSm&Nw*f#AQO)$Jruv8!1YZEEz0bId$Sez$2Dr%3YF^{<#9G&G zw_B}NSkEa^!5CB#vFi*O)&*y>&xSvQ+L}jXc#K9S_UtuLzu&)k{rdXdI|mM|0J|#q zIJdfy-$(Ts_bo78T*@J!AfYzSmU&qUAd+&8d@V{tk`U-IP7bJd0VI$UEmer>!=S z*;(Jbd+9o&AyisRx7+Hk^wK2Gr`y@ihzM+6P#{Q@)+DKk(rjxxNzCHsJJTE+N*UDs zQr`1|A$t)uQ_nVqt0Oj7DZM~UTf9yePkRLz^0v$%2-Z8x&KNCFHpeZ@yhD^YZMQF9 zy>{iw73Z9@<;fEd9zJy7fn$e8qfPI9Q4|PD>o`f`wSxy9f8w!s-~Zs)@dqA!@BznA zI-j;X)5*}Y7)A9>L+U^kA_$Qk^pjrqfkSIcCypPzasBq@=606lTI;2y!O^3K4uTD$&<~P22@W=y|z3&k6wTO`DuRZ z+O?BUJ+|`D5z|*gRHwf4rOu(1cYgZf<_mAQJ43)IZncW62to%cB}Nlt;eGB)Ffr`j zA7PzSs3LG86`kX^3A20-jq)LZ5r#?(bs@6=0Z^HjH9FLLwpIVRpst^_p1osqLLSg@ zg#(yVF+$Dt=lpVDz5r~b)z zj(y==p3tVv4h+(bYug|E;^jC0?&t8(|_mbJ~&F)!xDK%6G?+N@?mPQW?y_hIqRL}{C zg}fJWjNk|zFc3l#f!4wTg;kajBsN(344{OzhJ%NrN6(!7;uju%##F{X$dBgSw&twDqsO?7p_XKY$B?i*7Weo{uVsQ5*RW_RhMoPgLnKJ zma&T~Y5E6^=aYqY2ag~G96oI4NNFu7&IflI18}S)7I^5PXaD%y$4?(!9~E&4AHVVL zYv21DxVsIR1q@{s0$w~Z5&7TIy>j0&1ocTm|2=;zOc592j^P9*08)Y^K|KjhF(fQE zMq!QY02~n*O-3Wp&t7|{ax@zaswP&WK$(G7U`!Op?RI;4dHK-V!NFi~;J^W`brQ!1 z+WiL(918RsF>3IC>g5bY4tbVOr_+3z*|PM`jdw=Y+H5l284kBLH#gQdhQpzo=CK1P zU0#-DS=zh|KzRVrNm})ufvUa3B#4f|F^GoDgAxD)ufQwF8IwVrZ`}pWt52VM;44qM zRg|61ox7u7zxZpobP3Wy!P9b@W2;*@G5rppco4^)S;Sd%?2sikY`V~jRhDTPF7 zn$AG&#%PU*pa3`z;9c+R?8VI4bSf-`wUg;I&+{VBLlWZd-Me{FOeT{&%L3-U$a7nk zq53M|gJwlTfDK2bRz3s4UwZA1Fr(){UQ-qEAJQlR24)VkI6_3x2;euzz(OKn@2aTk zSo66S$XB%53sP2AbBaW92w?wtcnN(vMg&Pm=H^E zwOVPMynpc>g+xeNX-!&_R-_2%;OfCv((3p6%Y)^!51(#7@k9~>XfS=4T9`~=I-OR~ z`q8*3inlLbWcJqiqR7j<2weik)1lS?3+tS7PE;r%h_oUSwt$W}P$LmiOke~V*#mp<9@wMz zAdKSJ2Yg3B3pSg3rkT%q?x12wkjxSF=b5z9bOtsUK1Q$#t-=gAM=3Ps3Jqb@Y8OZx z`ab{#RrPE2BoRjO)lUT|pc!e=wt%wl+Jfuvu&eM#5QJuf{2I&+0)mkS5JeT|gb>3F z%qShsfA2gpt5B$^(b^IbfWnSJJctu!5Yjr(Zy1EZA`)pu1Z7!Nnu5Y=yNF0ciPB+O z;6;!?DHCbqHC@`g?Ts>tPSaMa-R}0f?M^3(%(-)Cwb4-=r)k=1rLA_m-BB$K*&X1_ z6?tBkWm%Nj*7lv9Q8u04x^dH4TV#2W=gv7_O?sd#Y^a+yq`HF0+M{=D1q zm@jpbDX}O86aX-tq2w!%O0aC# zIIa-%U=fMqI1CYOMf~#KdlnunErrn=hnY&X3Fo!76+|@JXccK~ltFC-Ri=Oxs_}4? z=k0d8V@xzIttN3!*xBI1PAO3VWg~2@OFJzPm1oxI^!SP6TAS6?gEwy69gnA3Hi?sH z_27Y%Cm(wF%&Gq2gOE>Qk7$?>l2V#koOOY$AR^Yft*xyr%bB?Wv?3xY4$ZpMpe0}}y#fujKP&{vKZXG$iHknM196n%73v2!| zGj}@ebLY-pxqjsrzxahQ`q04x-WFlyq8bb*&NS%omEmwZijyQs`-6U`(|h27<7MHz zXAx<&(pEb`l8?Y0gxpSt)lMWfJo##`4WEq{I6em^_{PN;}0)f zc%;+mgurF1)j}d`t>;N_pd(ct5U6Y)s$5_1_WEg(jD|ZezVO0iGO=ZO=gyrxFY=-& ztrf&pyGx|AX?AyWb7CzZve8~EOA^K@p2UuZvjMmw(&0W!+5-u4vs9|n0dATxGm0Rh z(p25tP|$juRV!EVQ$(C>Zll&FGRhc0^up}fvj^otVYSuLDik;YA*4wlAp;@Aad&BO zU~Tn4uXpC~;SNFq{?ePT{`9Z@V*BcQ8{7K$!LHKj!2@e04D+F-qHF=A?6)`Wj&9mX zHrd$Ryt}@!zP_=!xj7l_K$e4JKm|mA3Iu=!Trj=)28&vq&pw=g88{CtAOxg<2sp4} zImC1wi{hH6g#v{GRIokCUcLC%jqBGgU3uawUp#u@O#gUy%k+9GgTpSk+XTfhGI zKP+B-uNdtln5YtC2GlaKwnQkk0t3cZu{K6<8U!#L^m;soA>52}m;<~ol}bKm0isdM z7b#m%deB;H?R`(6zCKSY(;P_d6_n-2MpFx90!OvfG zlu!F3%4`pXbL09@j?n0LO>3k8q7xJ}thAs)L_#LCN^)0#F~A;1J03upB#O0jF1I;o z4J*CQqYppwhhIDKg-1IF`murcU;XID#Sh;8$qWAFH(_H0CZ(hkQ|b$bYGz*jFoC&j zRMcqq2pD{^jF9b27CmFN!7PgAJYrb;31GivF7AaWX<|Y5Gm!ZG!}ZBbzYwZ>*&L%H z;{XIvkrj3#U;viVul%q6#i27N?`}>8?e@*Lum0#Sz6V!s0Y?BDg*1Z$CF0n5yx-Px zx&N`N!2az;GYvWzjtEf)r##e%(mKN8oD-}9UDf$bq==SHOFT0&GYbIv%FWBBQ=tP& z@#38?nOVSh*H*GJccZ-A$;!>q_Vqg-$=l*NoIkY4+_xGz)zD5lgTczm%7K*wE6dBh zUN4E_-ttn&lBaQ!#BmfwT9W~AqmoNoK^yWSG+v6xrAwErwUhC9dwY9xb8|QxPN!36 zooTg$HdcCOW-kol186jtDI8r+A3Jm5yI+;n=+bs3rF;FyKajh(AT^L`n1J-+qLUVR zVeJ?+fmRS{PzsSER1rbpb1)i=F;NtV^0A4GF$WJGjEqT=q}6J5I-ORll_ZJM+7lp= zF(!&46VJ*~=XL6O6C#Xo_4*C23k*E7wI!6oA|GG9;+)GD9pyeNa9S!aP?vu;#j1WX z1)`vjH$A|@l9(vu;b(;#BAPjlVgv*skV?HEBoZjb6BSG$NJ>Sdh%^$Q&B~Cvz?yLw ze1FS<*leFkNt|V*4 zqC`{(KUN;#nSI&|IQEv=0|=2IbUWQKgGNOn;63X~#`peiFZv4C@~w52auDQ%2tciT(-{_^tjU}?GEALv$cq+87;2 zU^Hl5?Kkh-vfjDrbh@)Xys>d*V`F1`d&`dtHTJ3$?|twQ1Ar|`Tb7YFRlEQoBGNk2 zN)aF^1_1VeRT^$s*RS71L_$rZh!iS7?UAywypvBd1=MSI2faRfDN7r*jQ1s1gb73}ghfY2 zYXd;Q3_x0uEpzAm;Ufo+9y_+PGs&_%&od*LTWiBNvDN}WD6h-&{H2#(>U28KJo5|y6h+bR_d}sv1K((P zQ1b%xybS#rexA{2lx11B+f_*d)*&iD4BB_)g?-)ZUoup=L{xjtRz=yr>((Y zFquq%`5(5+VV{@QI*#LUO>YQ~;51G1JRc5+h*%Z{k|t#Y#d{`2pUsa%Eu zM$8;UQ}*0M<$y&nEV?CxX?Gp_@Ld>52w4b_7%|K9$#mN7L{O>E^>(kmTRB$(u=hTW zQU=*rA8v0AH#WAu_{js zcY=tD85-Ii)`oBvK_Ww8dM@7kaK5tgO$~8~*oND%Uhal+6t4cRHd?Dll@Veq?smI_ zrKRNqD}&{wcDEZPag@YKtJUlGTJ5%uOz^xa&uQt}Aj%eQveRE#-MaM7)^L2VpOVkT zF*)zY<2Qft{Iy#X7atgpor+WMf~+icR(=(Z~mW@RxEP=EyD zeZ7AM@IYK^6pES_h;Y9Mn+gr!tbj2<3T_RD8^8Sd^}7!|ap~c2J$vB9vF@6gMCRaA z=f8bu?ZcnH{N9gV$gge2g-@B4^P1TauvUOdm7*9Np-1pE>z+nIVj-Co_Bg7#)=+-0 zBnX*@oJz%*B2WlQ6DdX{QlvEyg7dnTDOkjdcK{B+212-^x`0h)sf^Fn1Wk;@*syyY z$?VpwRGv=(#l(?`jb)#gIba!o`Ro_}$seBi>eD+B?{MDhre?@jUVP)%|F3@oS2kdM zn)nDyKOT)hsqSDf$+NjKu7zTe|MB(rXw-y-(Jl(4#sg$*b9A(}eE3U`ocr@{oPYL0 zs^Rk6x8MBHFE0M@d3g5{ui_87U12tKEC?iM=$@I@4?m-Hb#(%?i^_A*q62#O`|9ZwA=;nwg?vm*4;ic z01yJjhB3_7t9?j#%sA(&eU!|T+7L)u{S_6`^8A z7Eq9R2TGvVUjEKk&wTS)uZgp1c6aC3|KW%B<+s7>R1^wY=h<1KG?UsB>hkHX0Q{84 zKa~0Gjpk?`Lo&l05nzN=2lGJYjHBIlH*}bI@16HyEv3EK$(W?hJ%pAs07MMTdL%+b zLu3dM(W4$+y$w{`d&r=NvM60y>L{AkZec||Rs@hc-YLenb|!G;a??U70!0BB9M&d^ z<2Yc5jXr+zM4Tkuey`u}cRHPFzfJhXuRM#jFM)#dRBKIi?c-~nCG+L9EJoAGcsibB z)7+JpZ(rYZ+2G{SqmQ499$3BWcx;P)t9|S0M{xWQJa}v==`NdewKEvBdw|$Gu*?c2 zSdmE?Bm<-YjvhH&!w@-q7D0ZEqbRUTi=rrlU)x&ey-~_pw=IB01d4Eye~^|UXuw9Vh+^ohSeTLyK)JLs2E}(@Qdm*vxqDY2Ef9exvaikIq7_% zh)k_YBcdinWF&JnqzNZx~yJ-4r{H| z6`+LX3naqBqV%~f$_yri6u3{Vq@|UPbQDKX93@E{MY>I-rsD!eE|=l0OT$}oiCNfr zVR6>SQPfIWop!t1?RMJTPP=VVefr^ZqQw|vBCQQn$q9jLH|}P6J{;|AY;J6BZEkLD zZ;y6to|8uS|sT4;P=|pQ%2;!JaZ`rvxj>0w$0mIvL z;1z>$palexkvxRjy6}&9U=QL^*ZU9fD3-m13 zKrxaBu`H!04~kJSET!GM4q~Q=NizVDctk+MsHy}%GKiok5dd(DOgGv82`&IyB=#I= z)KYb#)O9ox`ud=*e5|VkY%nPDC9;2oQI--qHt3B zTOVBI!sXL!w6inZ8tx2-lbunK<;bFuA{|9qhqXuQ89bmv^uhBZh@KErFd$kvYI?Rw zP;8KbeN2j(t#{&_@nRH_Nm?ZAP+8={7{x6eCu~{70bp;X+vB!)j|32D&?MP(XEf>$dJB%9dO69==sW;wrAce> zJe^FoH@CcV2%&qM78WO;NlOji#Csn_QIaIy`*+`c_s*R=Pd@qN*|TShqCmtzRcf^6 z3TwXj`t1d1Yi(JU#+ZtGt94^%t~8)I0lKM-{scR4BS0&q5W!j-8Dq;rM2;OjYP8Cx z)3n|GPhl4)GXLI{Qmr)ILexqDKv@=w$Y_0gIJ|x9*4n{Cp@#s8#x7=t^4LTzVYmrpY8Sg<7`sqMK;ZaIf;_A)e%7NENLxsUA;ARpz8My z86yBF%hG!XyXsax*R8QEb_`0bLD%TAU<51+pTwy!Zrt5^ULVj z7;DR|t?|~@d)Kbsx_xWo_N~n?e(}kqy=3!oS>&yD2hbP!RI?)7yQ2jVD+CPyh}7)+ zeQ}Z^5-2+Lr7DCZA|%jMI5zHqfG1l-mR8R=4+ts}R-VBzPm6q#O+0t&>Li-mP#~ZX z;b5s}jPYJjRPfyox3d>seEq2}Ja+cXNn=t6r7)AiC{kz6Jbd%!&5IY`dhu5;e(B3k zM^P-|1%hHOa}VUIJXe_wa7~3p!0H~B=_9AQ|WlV%5&ROSd<$i8ZTc`jv={QRI z(;PsxFMWLThd=q%+Tn-(?9cz1XKlp;f>O#v5h?P_AsiTD1Q7T_Qu(+Er4>cqdlJ@! zJKNiXZkG^4%0?N}?e^QfUbou^pq-uZ`t9|YqA5DfNL~a0mC6ZGK)k0a&cXrG9oR;> z%R(1aFezhw2WCd4lpuL#_6`756j5rJENBQ`5YQTw60Lz~Wo@n3>#eS?t{gbf>-Ew! zHIX5s`)jKlmJ$^oFH|7nom;oTv)>%AUtb??ZQr?l`_`>n+uK`rZe52WQwXuor#Eka zijl~9SZ?>Q&0chF&Fyiyefg4E=GgRg#_WSjxH6iB6V8pMjfkm{6e_J06Hd2ARf4n0 zVG{|>@jnq2A`uy-8PSSw9-311YzTrVB9~|2#Xy^*t}hCI@yf?n?_B%f<0t;$tLMJ_ zl1dNGjXr!JDo{X1X=+Cvj}rlWGxKfb+dp6;V?}MTXI2S5JCk=08~}U;pck? zCIQ6=0G`mJcnl6wA(2RvEj*BbmZ_frmvH>xkw;E{`+xb*d62q6)H0x*3_ra4#!r5E z`LBKe*EXS`$ddMsS)wGdipI7StO|MNg&g@G5gdO%=O%(z2vtW<{Xk~{kf_rtM>~)} za`ezcUwZ6`KmNwrV<%k+@4tHKep>GU&Y|>X_GbZhex7_?l}00`DsI5&9z=wRy@*KZ z!Iq|#L}@y)Q^>YqWsp4l@Z;b5>gmTWoIH8l7}&Tme)FZ*Ui#_t!*@P_+gkuyfk||% z82lu+rDf|#nfg9o?dJpuL`-6Kffs@R?J5OCm}@{=u$L)4*VRl5QH?!6Uj9#Xu!D0H z(fM>?@QZl{VGroZ!%8py|Fid>PnuoVnJ>7^-shaxr7a5-T)}%!kf4=F%C*$px4UC{ z;!aO=#C)F*6EQLW&X_wlCZ_Mb-BL?xQ4)=4L4Y6#@5`Z3HvMak+k3AyANDz~$;<*! z5UCb5s$digSy`EHp7)%+*M8Qse$S)#eBD2FrCaPJ0XS2 z`2b+MmW`3%5I@HTa5Miz5BLKY==V53^@)h2fGv|fYE05TyTy2dEz6kWv@o$j#Dusd z|LfeX#murWcB=s3Ro9o+1tEk`*L9ueOw`S?&hk>8=Tnz`vkVCWz<6iE1aSg>dLjh? zYr7&MPNb@JTvs&l%*y!kN0-15FcZfex(U0m-ENbNm!)Z%##Ft8>{AavZmgw2C+TG; zZaHaE%Z5C|*H;I7?7MgMw*j^)Plg62>v{UW{@?yD$posM3QN}m<^`A5Y4na8(^bD2qbduq|+IrX7cNNHf-~(WikJ>Ux&i zF+FMu_<9`iY&J6O*f0?y7L%g2sbUMIB?gqTAO+KW#1Io3Ld5arun~L#8Dl`QK|Hk{D(8L5jBG^)l(Y&mi{5M5oj&|YMj&e~6p7gL8DPH!1% zJiHXYNIW^jl*wM2)FdTHu&mvryT0NiFtBlQ>BDn?oFTIcDR}SWfu|9BkmB_TA(?jD zrfQ2pn(PdXHr+U;#`25`TRU6PAI?b3Y>hRWdE336PLgJjX21-4yAV8-Wic8K zx3)GnH#WAmw({|0JQ|8`t`aj5qKHU4Z+ZlNN?Q+k8PZjvw|ro<_-3!$#-qAFc| z%gXZAOILS3`1t7Qd#;R&Vr6j06A!)dy%)_yQ?ywHSq(-=6)d4c2&y1rlu#m~N=1|q zI%%dVL0i0i2&w`q%UMTNgM{GY9TCGA1uYW+BPPVa1~_h1NX!!dW&F$_3K~G1bAn1> z7}*d38D&rkN?B&o(IoiV2QNgFBxzYW#3-3Vgc6BJgj5i-HC6;9cmgCg!TCHdOk$^{ zE-Q0jB7PS@jyiH2ks+caNr-4?XXnKi zUp#i~*ne}BeWn1Xa}p?LSQyvu+I{)SWn9QP+e@9 zGI0oDny9(Dx>{Fdp68u@|2M|v{O{?2&ay0yM|kf+mD!BP<8x=vUi$db!;d`L>2!ci zyyuvi;>EoAw`{6D#uDFqSC%CqUA%a)+v(hY@4aPF{>5MX#pcGwx4-@EqsNXZ;G`(V zdqVNxGxaC`MKr?x@P%UisFWXRcn^Dog*|7oO~|t;l#= zMFE3|mnMT6_YwT2J7Uvq6N!^~ky>_uhJpYXE9B|u0Af4qfHa#=lXhqi=gCv^NJw-6 zY?7*{$~zal3$AufR5+Gh5y8=cu&~%f|@(1TG|EK@_=V*KX`rrKf)gvdaZ*HgE zE+86a1ud&eRZXM};|ymp&94Cf3Em)CB2pdg?H*ZQ55Z^L0pu*}^!kHduTR8vS#NLe zUb%YRMSFoNt4fP0RS-@LMd_r)*0VL57@wJ#4;TZPRAVZef`YdUX_h2Ol4XOzU}a?` zUFsdb_jX7*P19bl*YEfH{eHL8O%kpuZ%yQ=Lshx)cs!X*Tv=Ry``x;#M|;EV&8_Xt z&7H07@!n9&$|kAH3z+2KJTLBWp)ao%ryl zzdZNCtNHmWFe*U}tBF$BJaZ#{FIHg1NZx34vso|ZAJqn;t^6j_YPvZ^5rcOg0;s6? zy38Tez+{^8xQXAA00Qbk0F~vE}yvCP&Cwn`obzPNs^W=zKR#ypbAf=rf!? z{oq%>aL<>Ya;Mj?@9nM#ulw}MOK-mPm){+~@d4}>nPglCSJb6y((88mJ?|>n-G!_h z=hqtLtqrpHt!q2~HX;H9p=4?ITP0BnN+1EX-rE7F;F0_8|Hc>Zd-n0WAHA~i3+N{$_R?^p_6tWKK=9`fBWd2D<5AUCAB;A{)Jcm_J?p~tJ_%vSBeWJNs$?XAQ8%ZWbtom z9gEHI8^o_79G1YT((u}*uP>5OQ=0(cEXT|T??XTkk$u5B*bLuMeqVp9_!mY{j(hc5wQZBQ%%126(9Fu)zh@BYd&&-|<3t1?EzeSxhrm-jAQ`-}hb zH@uU(ac%-8K%sVZT?12DS3vEAnvwxz0YPx4VInmWbBi~D5CE`oIz$T}F@r%LV)%PZ zl4zj40Npgz(2THABin*1kVzXenUDYHL={xmdi{8NY7h+q0aS$)dTB2jmL-Td0SzLG z04qsnKLKAr6~@~`Xvv|5D5V4unW!o&lo&D+#kppMGl`Ao%a|2O(&(7RS``sh0+hCq zAOHj^ZGciN=M^COQ<_^^1)Jm(!5~YVN?j*e>QqYcA?xtkAU(2D5E=w6>knRk3zS0b zK-3Vk;n2FDz4z0Wq_sBcR?>Qr5Rs8qmj@UcXF&+T2dAzE(ZtyH&IUoL)j992gezCB zCTfi_#@K{SViIdpL#8e&L&mZ-Y$)?nX^Ck` zs7tEfyp7%B2FbvvjzgwPiqru6}B2w)M^!kB2zd*z)>Bm zppvQ)NTl&t=%N^{hvB~zr8L^SLexl_ZbD^sCPqgszB!vPL4cqbR0SbGB$Fr@RzsX7 zole#r^asmJOUuhyH#>dLoop~m(oWjxW!-LSQws^h_B9`BUl&!Cmy?gTuaCx)(ZrY4 zl?xZu2j^VO=LCrp612X$3S|X($yL?yGGI=UPR2G33^A0|1qn_=S&2HuZq`*V8k7_* zTWbs>DT1pTIIIdtKv2biM9|?*4C}*a{09(G5vYnho)K9!1I~;Yn2yQ1XRZycaTi{G z>#nEnIkqzJ7CN`CrMDm7Ja-{gtp&RQ5(gmQ5j#1r{6fV!B^b@^@bDe0NA;4m& z*x>@SDk#!)rw}1w7Hdqidbgu>RaJ$$<|I8>!Q*c;2AYVhwboiyQIUAheDu-9*I$2q zV`HP=?*kDm#zlUP2F9Fh*$jl#b)DzAs$RQxZ7}Hn;UE4%krzMs{`Wup@WWoW_w+MQ zA3b)wn{`Ix$#8Ekj!_X|(&@x|UJocZs>;@8SwXz9;bXy3UaNJT_8vEUB7zEO42X7*5Gp0)@h-AMikAvot$AwIm*2G!I-BqE|Av_vU9 zqzj7xz>P3 zHwG)KwX37?UPQh3+)NMD^Uw|aBrm1(XEOJ*@x12oL@AuQJ)9Ln-tkdiD6PqHE ztLr>3hU3Y|RX|3wM6$#qa}vF%CiozR6Kf2mu{1)aIw4I3`W+Z7t*)&fIdWuuZGCBJ zImNHrm*L z@fd;tGmr&`#zQPtp&5b{P<2y_92-Ukfe;*s2aA9V+%XsgQ&HoE6q6vxHp#MVZDo1l z^HGtz$&SZhvJ+lV6jU@Ge`9$^23s(jKzjxRMHbt*66P0tRhyc>&EoqusOex3w^61)ke)gAdzJL2ypMCfn&oAG8q)6b_-?=>; z8<@L}Uw!@kjW<4ki`PTt$(j%%?0A-m)t=^6TI~h_K>$J%$oY@Y1*bDBrM8|KiHVC! zq!KDuPXOvczhffHpBk#WKrC4mRs8gbk8PmJ=UQ+-kAAK(y<8PkS%FlYRLLN#F+eKj zk$ax`SKqq*3y;EbHkPooJdn-N3qStxTYvUlID5rNVhePmkU>JG> zoC*-FypI1ILii6Rl%qlK9fvMQ;1qcG^e(IXH4{+FNqqi=Qwo!yE5=*{R`{S41{q9fU!%I-=Ky0_W;tKDAM};g+jkO_!$#}fCxtXmjM>zlqr-9=Fiitr) zZOy9C%)oGe*EJ)y;4n}H2ns<5oC1Jy!&!p>Pz1%WOj-Oi z5fvwZ0Z>5w5~m_s1*t;_D8echAw?MO*V41yu3k?04his zqKyI(xoz+&Vq~*5W?sTJzyylXBmF?gkUWAbr6efy%hm=h+%(P-2-qKT$bg@O)x zgDH-KfIz@`SB22)bmMy@B7oYU2@%0Ew5(5QBxVQ^?Mft5I6Z)DN)R!!FuEg? zB$+Na(8kXpqOsOh6#{q(6R0H{@cL5cSifX22)?MtXD>hyU)3tiY&|R->8IS@x~}7o zYFZ9^jF=l2IRQ3fqKsrX_6J;56DolQkQk8`yr73A+jC0b0X(7r0TCKT9(4N_Z3;<( z2_X|ABa>3tts`d6qGe{Ph$;k-CG0e&ENz|wMD!F!<)$Nl3AFKh5-6h(B%PEJlXN@F zYpZKVkF2a6x#jrn3E|PNJfBiD#(}G0JT6A#QCSriE?g*zd^p_O9q#4%q^@hPp;W}o zLc4jj0}zVsa)&CUZEjzr?idrdJCyi-sR9t7_n~pIM1w=z_)r5WDoSf!7Hp;NI5etA zqUe;3%a$l z8CO*Zzf4!3sqwst1Fqk2+79ndRzC4qztDQZyl0lhF#hX|1kT8fTy#Qe-ABQKFL z+$TWwDpgSN2%d-&wrn!8s6jNSil&?@sFI>00clX@eF(m>%McNwiBm2OTpPqkmbt`k z9Ge6YeG)L*?F4L4^g*0=W0!U4jnE30<6Yj+a>>wCTmhSb~KyECr40ji`zb0t=z6oHqD-RFP)94@9+>bYi+jW3Lg7 zHB2bHqM|YYQ9N-GDR59U$PC7NnM{hZs+o-sYHX9o1B!&Cq7iY_>-F==gpEymeKr>A zTD(UkBqdfNLZ;w-RTs`$iz5$^k`L82gB~>Y0LY*CgM$-2huc+1nR1) ztWBbnhI1JP_?aCm*nq-XQiq5HVQXDo+r%PbQ4}g6M%#7f1nA#h^?BbPj4_7U*Dm;w zq?U-TUAuPf+_}kkOn~B?51RCB`wo8YK%*!R#O8!L=c=+OCKD#=cDsYY;GqW|Jbvuh z>#x54+H0?EZ*PC;OV2;>(1S~ZeqNMSnGbg$cz5E|ZH-k$b-&P!}x|XpG&sy7l1)XE(0z5GLKu5|MRv?!9{NoC^e;WdvZ; zq@N^adwcXZfBl0=KKlLN|JurW-#Z`<5&{7q0?C^t=rmZK5CSMLv-jc9hCuwBh<+HCT9sApd?gUk4lGqXO}EhER+keAkr7YC3Pu7kT4PL_ z;ngb_`n?Nx-+gAqNX%AMoU6-Xk{F8(LmCATiP2cJsrUSs-}&x4A71!(|Nj5@ z4Foz%)6^zaRf`Y(e*eDv?mKz%r1LUZ zUL|AGEF-o+%*3GRW5`yPgtZ=YdJ37h87PCz){h=tU0ynQ;@GjHN0yhDvrfmRiC7-o za>P+A&11>U(PFT%vE{ulieh(fI37=~U%!6o(xu_X=H%LS_927--osqPck^eFMy?eF zUqU?rM?q&}G&4+dbD%?Y5?q+6EtEiAfp=msp{z7rnT&UKC(E6|Xi_-uyS+|2tVWYu zL$SPeeBwa>W0?nt&_Lio01z3W4Q_!p7CJO~MQTwff8*&+k3) z@UOn}cjtcabGWo2rPNMHaOU-D`(uGnrx{lYMnJSRN!^SsQQ;p3uDj3Tg`f%&Otjrp zmHG*gTNx}F1PUSyN{s4bf;X!7;z7I%Sd&RP=`ymYYVbD-^_tfO4%Qi;@mCiO=J{f?f+q%58)E=bC3x?H2r>tfx~@SaO%jvXAin>0!nlmPHQ98m>cE2=Q+0y05aj6kRI3NxgoNfog`@I<38haccYKr?DKN7 zU*~WEH#)~KJ`jisEb2uCBuhJ!@uDc`^HX+u{SElmURW|r=I%av!%Fh zHyb$k-uHf1{OUv4u8mV0P`wv`BuOGy#HNWLHfT~*7GRdOrL)ru)`Y`OU>Q4NfW?eOW`m=h%yn;GY6FpUC$B-~=_vGo@VC}2Mm0H+Iw z83b%RjD*db8)LZ)*A2-Lkmq3c2;RFkM5Q5s$2+zRj$#X<1xG7I=Rs^D;Pk7pBJmVC z*L3gVTS-iLU7=wYd?ntj_4Y>FU;p<1VL}iPpw?<{45K`=sjnw&(yFMym^m(lY6}Zu zj(LZTA`SrYEZ6koRN_UAd|(hDL)u&cjkt6$1fUZf$C(J!wjZK|iaXT?mPnC~N5&ds z$RwJb$WB^GVjR0DU{F9JKu{&s#*5JeB!WTAyzx}EdXGQ&q)lzNpQRa)0p}pk{kX2L4<~u;#^dqc&hGB^ z_TJ9UWIT3d$#DQ)6jd3i0l6rKTGacXf`r6`#1ew@4h-4EvNftwRApI}QIbwD!63l} z1yGJrl?VX7stHZcnze+>jhz#x*RSnX-82o>SOaxF9+)JQb(-{sb%hW*X^#a})oV~z zuL?l~Adj*34ZZ~%oz6VUyoqZwFf!^*n@S19%+@BveC){4e!suIwzj&uI#^on z_xs&$FG-RkM>wu)-a%bJF)6FOklJm1bhVu1!=2sTt?lt}SmuTJpe;l^rY5o?2_}La zK_kS1rYW&760sGmVACR>mkkiGBLmX-=XQxuBc#D%0L6d`;G5|b0>UCXaCI(O!&)+I zG+^U))A?P+`x*d3HJ*swKlp=wtR)1xyZkl|*4 zB(7S3aZIb(6Hoz3RZvl4xFI%?l|e&QR0bFnFI~R!%;R_NUcYLS)Q7MzB_BT*4AyYF_5IcFaFBW(Tvq79eqT6u;0YI;sB#HNaZ*OmXecgNi+H0?U{PD;C@jw1Y zYi(6k@wj1aLqVMyBj$Nl%p5(%@p(bSIHVNEr9Yv*nkvIlTtK4r_4Rx2z4whb-l*#a z;@|1@s;X*6V~JwBU_ONK(61j~r}uuEl-LZxh}a~N;2UQsQ9hm=T|d&Qivrm2E#n9H zw`UmF7&HH~1As{qX0};pduw}{mh!y%$xnWg=lRj2M+bw!&i3~DiCbJ%DUwYxz4^mz z9FC4VVQ}6W0tv3Jf_K5Y>({Rt<}68zJU_C&{*^C3|K;bu^jF{c>z7~tMOl`QJoacO zwUfLkCgaI?R22E@+WN|oBZ%a^tDQ^J%n*YDGXdxwcibToE?>S}mPPB`p3=bg)xGTk zFyF4w88m$UOg+Vf<1~pxl^WMg{GZPL01N!_Spp$Y@G=@re)X%@LeRlrr7S&yG9?ND zP!pQ~ppQ|xn0P1M?#ghu@sl6F;9T+T-}~yZw^oY_xMurFKH`o`?Y;I9Jzo zll&Zlw!{W(7hGu)!|EUipR{jdM+ zJ8!)A-j}}e&HEpEc$DX9mMOuwm_#GSbUdQ2>j;=Ws2xR()>W>Xb^u^&d&k)1|iUX zL}ht6$-|^9Kln%{<=)=j=H}+c#^&bc*52N*EGj69MR-hmV|yPKPs}Y5QD*Ih8C;Zs zSYtNh!O*%{66oBrkUNuhd<`-wZ6~u!M@Hq|aI!tjlhw6BX3b=8lywsr<>mIZWN;LC z5XutLl#LBV1&F58^;H!o-LzpSNt1P@NJ4Q_29*$IS|4(ZPn3PBC%*?v$!;mWpMLp&_s5Su|Llu@ z{rzj-e*v}wg25o6sED}J>56DEo=~@6PI3S=HYKl#VoD|isi%(Ge^`fIlTCU+=WGm) zsEcB^8f(a;uG#^Zn2?8jJG1rrk}&4eyQ$YSxw+i}Uy{1W*_Z24z*%-XReilNfg1i)!s_vc?!z!ilQ@J@ivp9i*pkd+_PU zA9?196DLj>;>|19-+1|zH(&bI=DCmI+7>tg!%%u6%K9BpbuNzFL#y$jr~(23VCxCf z13yKKdDXc=l=c&+(pX@b%Az> z9_(daCorTKOt z2otIEsMN7Zu5~3ImO5Ykqi@^gt{o&A;LPjqfAs1*aCI*o`-}}ur--NGk;Z5^QdLx9 z08)tJD+}|CRyv9bNYjS#!VHc=@Z=1@0TP(toRR5?JSGQFkifyryt-|FU(iVH z1XvV+pqiA|Hm_bc*O6&&Jn|}nN=#`d>-M^xZU<9)&!Z0wY-j!0QgZ5!G)Yp6)&esW zIfNipS&T>H-JPAS&CQ*iopO>7w{{^giU1)n0RaR;A(H?dK;plcs;;VfA}ThuX>X7K zfI>0OEwM?hLGlE(hT7M`yUZF7>YWSDqKH9Am^DBhYDYmB0l}j$d}%uN*pcIx&s<1= z4I8xV36zi+1A%su8YG0^y#rN~+D@n6P1D`oT|rS&BI&i(m2?|AQhRr{(I=fMRQWr3be(FLiT zI*(p5K*P*Pk)j<%WYTEyj!HC<&eUgAB!z(BxFOQOf&d*UUO4}sO}9QsGqTirx0?qj zdVq0C|7uncktVHC58J!Rm&5ME*>W=Unx%N_ zL-(A!@X}h(LTx6y#Y)=sK7b0&$4|9SdOQLJT%7&Kx#tO|JL7cW!$a*rh@^8PJFV>_ zem+Gze>1PD2#S`NfC5NW)n!=+Q6vN4rxvp)WAX4KF9Vnmw;GuTHW&!d(?ClN9vO**Y z07`6Z9OKcZIHCwZi~m^^m9bft;l+zrUU=c<=fC{Sv0IK5!<~*bu;64@oad~zvwhRH z%JdP79@*Gs@jl!rN2#3-Y-UHM^X6%qG8<7O6+l*CQT0yHm?X(|w?BI8%{QtFPMx}C zWqGjFN$bg|0rX8TNOJ?UZ;ZyRr1@V^O{}S1xwkhQO(xd%V*B1Z*X{NvMd1TK_0*H2 z@%ZImzOu5mcE@d}(yUw8-aD6C@;(GnV@MC(fAElFca+AhHG9L6qBU8Pb~^o)wXMDU zhd=$rJ0G5X{OM<(d;ZIuB%|R3$$~(drI9}WsgM1%`NnMWqGUFRly$mUr{h(Fs7(`O z1_dGlM(5mQGAZ*M`b)=G*Kav`Y_Pt5d26zD?b>kskpf-(=%dMaw7b2%x4Rqi9BKza zpb9Xo*|+ql2AbUFDz#yfGlRS;L7Vrda|O1-^sSyhG6ce!Q4Y?y&ZidXLP0J9#-j?U z8d4H9$Qk$~FMUz_EG3f|V?DD5@#0G$4`iuH42P!B(0$QDIFE?fx2Vsb9QEJ{QK!e^ zxo5Sh9x#d!v3Q7RV-G4S(TkCmT8>I6FJ4qW|IYhQeCw+ZfAN|AvCP*{9bdWs>tDF* z{`-FMr{4=_FT>UzWL;L8Y;S>LZ)v#-4y++#RG_X)Cuq}-VRCX4qOATyQ9yMuW(`n5 zf@*CjSa6{pfsRie={^4N?Pk7cY^q2Q0HNj`R270?0GS9B8E>!$|62A=1n_0f%!y^2 z7zb_`T-5_u>7V}NZ=8DO;bRZqW0s6HU`p6{|NKXO7BWn1f-!ap{#9^u>!Rd zB_3mFmnKbT2<9}hg(?hjVPtta^F7l?8!Hel>;(rPo1a;0{w>y^vB~jnP@z$Pjs{z` zq#Xke;LhWZf9or^KXQ5-HB~(K@|&;y&5z){3zYPza3GL&v(neVG>dYMFb~AvxImAe z2ic_irq|~L=tEAw8b#~?Xh$iCR>v?fBtg)?inon5+%`zFx}T9 zky+aB74oKmhWR80(!_rkgyID^O$Eo_*_hxJ5nW`b42|mNSbD!%l$+n^U&N z&Y-y9L4)TC$uJTFTV>HW)|5XCu}X71g#;qYC4<|#2`R2jinqxH8|gBkfKw8 zG7vjr;kNEnbrPHe0e@okWF36vovS<)Snr9Wf_UjQV*n+V6eQr-0gOc9Xpn#X}{m^_j_3ev1>#EJC_Rb zT37XCYkX~QW3;JY8=!Bgg_eF@lcW0k%y(1Vv^8fB~c|t4Usn0$UrZvf)8$qkE&qvQtTAyug864tu2aznY-QY<;$1feDh5rdh*F9 zt+i8YRkSn5`GC)Ie)%l)=c$2o{>US$D=RB^-+lMH@4ffI2Om88)YAY^RaLj&pABvv z08M4fJnSU*c>)P> z>W@LyIp^XHZ!OgueJ}6*#>Pfbl!iJX2%=#&wTt6i+G@}XL=9#O09P(=42QcbD=RB2 zYY34W7TJnA6QJ8;`wT8O&HbjhXv`iu?|oI(hjm{Bfnr{Tw&%8~6i^hUO;V=B*oLle ztkFP7$(aw%zWCyc*RBp9dHB?G&pu@hs(6CNApujzNkid^dEbmkBB}@y3H}5?Se5nO za99+DE_bm>ZS#O0QOc@1b^Be9KK|&3XV1R;@-J6bR!$r{T1-aXOO_iBq?2ZtWp+r`wrKa_`;#Tw-mg zfb;MF-13cVOi>goYipCTRK%>;145J7B+FvB7BK_BWHR2`+N^hmkjvRW`~LsujmtG= zyQN~%cft@9jnTS>s)7&z8Q4tIaKhoCR5}%%X96^+O#L=E=S^wgy=~W~{evSE)4mUj z_ccoN;xrDAR`^G+z+P2Zfi{4n?LU4(bacP75Rw0YaZ z_nx}%^vPr0!Qj69wqp6ju{VG6i%UO#2`+6qFTGyJIFLz6AsAzfwPG+>Qh^Bt@wGJ! z|JX994t^j8D@tj}87azq!@%g?J8t{tS06X9U_)pQ^#B5qhNqzDAaGpJrrCX;u>*Yb zBQ^@fg!L8*-rnOs49}ks(mYzScVW4``0OFhfUT6pX+E?YIf_@5j-+JVkhadfeuRr|U zJyiwozH$EiPhYw4;%ji`DuyIwQx3M|#+tEf2>RbeA-iD- zakJrvH~Ajim`R2=s>7k}zt0p3grt+9H9$`FG!W^XD&r_={h`#Vg>1xVuzp8qnpniUVsoO$o>eV02ju0AL^x z1W!N;NtkXn?Y`Cw-_E8cVeTv!aQfQ!d8DSwSqm14JKzl`9{ve(-{W{QC`gQ$R3%ar zRYJ|IM-TU&y6>CMpM2mnCIG|A@8m!E^Y4`BE`f07u_|k}2{J=15KTbRNTCWSwAc># z4fJ?o$~@le0~tXAb32aQpi4xcEVKI!4{OXU_}LIY;RM`}2>0___6xYW-!i@MeFM!X zJ$U#K!(y?6#7P3;0z+xQd50&DKmNO4aTy2JP{PG`KDheY+c2*BoF<-}N-IXTbcyyQ zq9Or;Xq(;!&Hv*eb%bE^D9vyGip}u;!9_Cx*BlD)h57>-xSoXnMd}e7`_% z(0e%ai^ocrby0vmtn`N#J;*vG47AqlxzF6U3q*m8m_q5$q(DF?j&;_S9cm&9PAUtCS;5f1Azhxq5!q97gP{Ml9)!00EkRV zh?amU-z^NWVHnE+Oo7Qkxh|vy`)b(pg?wSv|78 zd~9uLb!EBN=QQc8FJlT>3WyLKR29^ts4PL)?5U%R-sySux+v$wZ5nM|PY-2yW) zOa!XIhv1wz0ntvH86RxKbCY4V2q`G&PCz8Yl&M3}cIzW^(L!{@K@7N806@`1EzPwm zqK|nNk2O_9TRajfvMhA`nx|72J4A>IKruP7eOz>Y@^;ZxJm;1x}~<5YC#CI`jMJ{Co!Bf?&O{12p5Qjg4?Nht|p~)3OH<5!eSQE2l_old2F_A)U3@ z8)t5#1cjm~V<2d!1IT39&QgF{Z+HBTCb8nx`+x{c9D&Zk`_XVX8jg#+AZBB@;X)7w z{q@MB#+c!7xV*ePnM_`N^;PHGlTSXmva(WDRg}7-1Nue;xqYO>X_(=sYS1w|vC*AH z8jgoCOG`^nKKbNNe)`k*-+v#R1Doy>-KH`$&QJ@V$>=H~(*c#XiHlw}3ef638ry2n ziug~W9#v59j~qF2?dsLdjg4oXe)>ya{2~Cc^{v8mQ;Yq z7?co#gn)oWQ4~dy9JdX57$k0HlRWQqJGY*?b+FXm*x0yq>0^;_?AY;1o(Dkh0|GLe zJ}k4b5iJGG>OC`cyWOfPcXqZx6$t;H>htWMsA!``G~|sb9j2)v8qn;g$YQ&@yLDZs zy+Q4S35i&>MwnK4^TmXwY3hAhRnB?c+s%LV%3CYT>yJNnKWN^7=;pd6*x;cww28;D zG^C~;o+b#Xgy5a4%c}MG(_H5U6huO3(oJx|NV>GNtUv&!u0v5eBw(@<@Z*m!{rE>e zzP@?wu}ALw#_xRf|4HOZ1(XUG?5ZICrg%6GxxxxF! z-=!=@#-!w7XLswi+iv^9bI-l{>TB=5_x|$IGMj|iGg}QFFoYmX4DiWGPB;sP0}#+8 z=?d7Tm7|>We)O|fUU>P9TW-7i8{hiFTW-5+G^z5k9t@Ur22;}+((1qp)_m!P+|nG# zfK9hQxW2WOmz7OZlcY9H(O6e|KxIP!u(Pwhv2nf5^R>Z=pr*dM<*8#a<1Ddc%hE~h z05CCZO~)7uK$Z7(2x_?Xlw(Ue22}Lkw<4>+l-aVNIm2Pruzj43xsgU)kaYJMP}(Lf z5`YFlML-VX0S((sLROS}lcF9BvLs2eRIb7oSVW^de>a4{Y715N-9kbn=On5CUUL2A%!P`-8ctq;$>ckdm~eDiA$ zKKn$v)`LN(^Vprsw;sKG|6Ol??}hDOyfwK#%rNU`9jmtR!Mi#TfiN5Amo-3E{VNDv6;|8X=cAcCrZK-CZuiEb3~ zb|@zJra`D6{pEh)irqYiu7%?(E05j#(6_&Q=W~zkpfeeD2+Hf@^FMm!z3=_3dgnY$ z94nEc5=lTO0-}hhAUVZ(0KfCgGzko|yFmYMh<%GaXFl`vGwiD}uQMd^Zb<9Q@1A%`C*X7MVjW-Ai zKfJ=$Pmr=dH5K|ZAvhG8VtQW3RRzoBy?6;|7z_a#Fvj+}oN-xQ@s3ZJIa1Qs= zPPxJXV8V>D+HP^tBG*(Npa+s3+w)y}hHVCY=aC4X`mVPF_5WBxJ8#+njaE?+YJ`C5 z0a$?v9jQPK-Q>2fJ^R$}e?3|4hYE0W_}2Gd*!$%h;5aoIR4ySS5kr=wAOZ>`zyhck z5QG2_NN$#@ag%5oMAe(i)Z9<8pZnk;8K|@F`e(Q!wB=BIHZ@%kln$}VDBkeNrNf|= zIz4_I_U!}!jBzI9=%8;`A3puS*T1lfVRdydx-xw4XD`FKOMsRH@}e-Q71XJy1Z~5a z08ttU!?ev4p{9qvTG7S0_*UqlaQKTPJk(7Nffg?z&U>l%Utm_SzOiIX*+C0me^C|@ zgaGq;_#r1n+IJF~)6QmLC{sU&%s}#pG&L99h{LonggA!~Y4XRIPTh4!+V4(K3{tf_ znq0b;)?x)vV@6&CT;4?4hzoc6;*YKevxP{(P8$%Y?azsg7n7#BFj3jjpuQjtgh)V0 z#DL6%isI@J!L@;$BE?8F<)PGoh5-|{Dch8-1;c_%S&tAcvjsND3}}!L*_S~TL*$W5 z%n~vXBt@eF5)=@SI7L>r2Ge9TVI@TZG=cz4Ixe`4I$mr>=&<3W!^UVr8{?syl*8?E z`0@4y{7_H@K|qSSG>J*F#H2RsW`m`{>gsB?vUJBi_Y#`{n=}Q@ATZPjUf|ug-tm3@ zh(;fEtCffE8ZEQ$QR-mnmZM+(m;d6l?;Y`$N?BcAqpXu9_IMw(f~tUV0ejUXFN)2w zq;WBN^Q;?}dwYAM;V_>}@=0EmrFZVwk)z^6)KfGF0Cb5G;WXD&m58}XKkCF#P#FuO zB_ItVUgEKk09{dWOEHKxF&~0gG1dxbi0>>B5g`#73bnSBod(gj@jP_LO=Hv+V9c=M;9kWP?Z-)EIxI+Fd|5(iUCPL0g(t$px^24PDZO) zwiWWt4?oJUj#iEic5=`R?s@9rSKd269_2>{{d_C$a6)9TNuI~4=Z&CoL$~6f&h0p= zvyr1&%{D)*H9Ly8{;&84J+R+3@|MixX+#VXsmxXYD(4iDto0I!N3yR>Lk3RZnRaNuJtFbnYD)1-B%6QnX8t51n=sqs;VkO z0z>hhP*uQx3J$5^-f(4k`QwWh2ZO=A_ui8v$*D9a*pT|gxWFp;X-G%2bIt1`{9+i$jm@mdoC~|Vdq4l_&&PYCFFgIsV^2K1 zvf3{vV@ByDw)QTH--PHTx){m_9dfgck2ShWt82qi(OX*S4UYWs z^>@Gf{hucN^*{Nee|_rCd!3LgtHIKmgfJdYjIq|*vZ?{5Csu9UA=qvrINe0zucEB` zOG~|_<<0F~CAMjYlLV6_X6(fdqj?+l80il2WbxkY`?2M zIOprSMCSlm<7vFC;wlV88XJ3rglMPWEuc2D8Takrc-Vn!{)1M~c$`j%>a@ma{~!c@ zP6t2*4uBGnNtHYxlc6Ldp9ICCti2E2Ea@2APm&4dRgwE%VLE+;pb{tsmxO>o({dPR z>z#h?(W6J#&)s(F!=Jt~K69zehil7gnmbidL5NXINeY;$lh1x0 z{|C@E=6Q_meI1X*9>J-bxO~e%ar?>azSBolmI?zrbG&))m|Q)~C#;!;>Kw%84qKRAE>#n(Rm*=z9O$1wI?vaa^P zM8pB3x1UwfD^Q!EM9`8q{#knd#5Zf&=lsn%poMTe+=jlvJ8yaDtIs|AN8dbg-}2Qh zJ^QQoU-++oQ@{EiT;4U1G8^zJq^8>sDs!;Zxw-7-ca*7rAi!_P!TtBs+;3tnXi&yC z`Lyfd2#!x%gm9sl8W*|N6%b)3HHmdq39feK2<|(za@Va7Kl9|{&pmtU^jcZLhwofC z|IP=m{NN|$)h)QP1%)>@>)OoMuAUT-q-1QOYzP!-GMYdT?)4EtW)w&uZIn$#@O025 zdWil70D@Bi`T%BldK8}f=@oAF-l(s0!aWCkT_&OF?IUWDzd-FRh_11 z-waR>Ky+$=nkLud4b-|rp&I*MrrzjD)x*KT^Ky(V3ZOSt3Sg5Fy1s7LzR0b>QK zEO6hQr@!!6@0RZN<|qNU`1-qhFTDw2on(h;s7caQtAxo6rpuyfc6866-q>B>4L5|zL8gYK9#KS^S+Aq?0i>c|% z@AGNS4BB(Q3qlMBWTMhJu=cLgr-1N5ebv~p~ z_j?@|ETz-?YmPY{uN%|+>yLBI3V|Tz{6d>qLxLKOND<^BtrZDT6{MD-DxrcAHl8go zE0Iv&cMWoaHes8vwPXyj0mDoT-1I&WRD)Lmg~TQzia}LGoDV^wa@0*zR+U(Zi%Jkc zK$Xm70UDqH1W*Tn;wdb1>PZ1XR0Tv-0!jclx_q>9RXwU@RD~prw#wnzYZc1%YJzNf zWYR2!ZXy|ZRv~%*OV7GK=c%RRhF6y^LyZKeaLanv-u1xA5t+?=GTPg_va>b%XuLPR z^5F%nMSQ5eFVqJSR?R%FX-9KcmL-U$NwQ@7CSAq^b}#RwY!Yh|(`6zM4Rt`TiJ(!6 zO0a4~NfRMX#yJ_2TEj%j01C{U7^9|7@%RrQpo$!ai_HS^hLzC@GK%sVb0? z%4`4vNl*)yxnwoTKDx5Caqi>W9(d3i7*%llWB0xHcP|t>mxM8Zv&f7smWg5wtkZ)r zf|6o8bF8ggG_5oirgqw9B`z4-7B%ShUtyy_QRGy{&!kx1#C{RaG55di0Gq-Z*#e+=&w>?z`{4G)?n7j{&)#U45R107M7$XFrS>W0EB4 z^?Kc2KTXrY(#qX;-~E?=^;hq_^UldzZ|n8@MOnsJu)`1>Hw=rOsn1h0X%xHBP+Ob? z-~1}e(%OWLZU61Uee&D;tK02fICHruijyZ#W?43wOr|sy?_61yY}2gU0|G?!K78I0 zas>eKzA8&s)rjDntLwVc=`;auDgdOabzS!ceede=Xp~sf$ud;|MNnZjNGKspM&sa> zFt%C28V1lbO_r7h=g!qzTU#Ln(_q2>zUuQ-Bc9$@QO-LYi+f`&B__rvNurG(9H&8c z(*+)<*k#qcC|6gOYiGT$4QIQf;{3VG@4b8WnJ4cglGcAaUD_J>hJ=Mtfaz#qgMN0w z)wTD2YH5nDa}jC2hUm-^I&JPFDBxsLj7NEz^_Ths6&_E@GiT1f`m5Ksw{{rcdYd)eNo==C$U*<>=YY43?Ap8V54{nK~edH2ZL+L6^|S5A7JB!Gkv7GhXFO@jtx z!nE7#^aty2y>sR-|J(O%w)Cyv`{Ud1yx)Tn1!nAY(z-4UTWigDGKvOPopPBX!LM=K zf&x+-=thVFh{S{C6}Gmhs`X>XI{iK{H>PzWBm}|nWPJ7N)xEvFZnvWq0qS@%&P<1G z>e+Prz1oSdoW`JKHf$1WQzQ#YaXXLUt|0`Kpdkjm?swEgDJDip-0)DdxP!+q32w*= zjW&tXwsbS;KC4_KB}qeIWF&GPkeQMcT>{8uRo9h+PG%ANNjkB%@}aH^>I#7OB4jcJ z4HD*E!qYKSY#q>_6zZk>*(*Z(w?bU-!Qj5;yi}f?60>juBEba)V8!4hOV~5{q6+U{ z_;6!u_sX>k7azXoo6l-5$$IID2k$(2>&erPKlIA?e|r9>FK=&egA0)LI$7UU-d8n( zOHHyrCh;H9FwE?2L=c3);KfbqVt1v3Pd*2& zm^iyBPv1=mnx1s11>AY`$m0*(`~1_#AAex&j@5v$dwK7O>0Eg6wb#D;lgaDv!l(jH zuwzOmY%L=TQnR`lfB+$cz}!fdtx>CSB=J9EpbzWVtNx~09}yP{DK9_y-~(TO?ye^v zSw6Kq9>Rr}-+1!}FVru*0lOtp)=#=M)cIrrRSgK#N#+Ni1)JoVX6tElj!b|g001~X zoB{gZ#lfmGa?&9&8iy{|Ous3EJUwPy0_x|>WaA^}bQxR?1VT1@kQachb0TK_S;GD)qwdc}9r=Q{&JRMMY=NO0k z1Z4vY4)=K%(4mjF%rRxAMw+>23Ncs=tasURq!s#+pf*EZ?i ze*5i3P`1dTn;%~R4_x_9lGXJj?Pm4oaD|D`fpvY~iD2&S5hL!<(*Y^XEFG%ST9`O# zQp|j7oEBA3iHvvz!nR97NT@*KK?DFjpa6Ry6a>U^H6dU|W5^gXh7A!}r6rRkz=qk9 zVPZxmU}^nEZYPbl+=R=?Rf>DxK3dcjSUFuL%i5%=dMjED$ZfE3`)=-AOS=W zT%M;O386`hRSOPk7mW9htsM`}VP1!ktE>wdNk(e)XaDn`Ry`OUOZs29``C%4q~}?C z-4Z|iyAO+(&heQ|vymqr2i+xySQquO7~>TM1jwOP1f#%0R0DcD085NU87q(w#QD>` zlioR(SFWm5#1N;(uuYISzAGZ4DnT@WBI00eJ<9M^2(|a#`w#+%3_2ZD4I&}N&{L$U zps^O3K}b;%0Z|PKnq-O2cPyqW!%S4wnd1i{8F?-=7^f)bHes^Cr?(>oVJgc`2VNBb z%)+Cua|e%E(`}~P-Vo?7iKsLfb5&}4F$$royKF=0x|U%wx%|QT)o1SGqnslgzomQX zp}WstxHg;=y{>hlK6qdTRZ%2WQjL}yne*n%`C%6+95WeR=c^CYDUzuPSDaffX5fuk z;ew=73ppSgONOP=NuKWx_sXj3cTJjRAs^}N%!%4a*C-;)hRLXS=Y3I>#!M8j(;tA+ z%#uB~YBC!H4$e;|`Di#9jq|#8*2Hj1j-oS>QJ@e4lKBMp1Kdmk;e`tqKKS5+!C>&v zLl5=){jw~hXgEzEY&0yyi1Gb|lgq&z!W(s$+ekxY<}^*a-EOzrP0~yO>bhQDUcP+! z^6Rg^{?s#HSU-0BK=C<4aD3`%cm7|Ah>Em06%6e!5vi-ncRZ(aKMwp;@Il)oIe-2< z5v{DOlx4|G%p5{cQ4z`WymmnW2g@sP;Eww2Y!fB`)vBt-lSx&Vu_Gx#I-QO+CL&bg z__ndOC<*{wT3J@_uV25Grs;k6-M6*5O)cVFLF%d+?G2p|gXMwEQdPCq_If?%+}_@v zbIv5muf_YkudA`|OxPl3N;lX0Yd94xi$ej`OCTa569Nz= zdG6nR_skQI-EKgLsTq8lt-Z5f^VsyMK`Mk$RaITrA%sJDo-uoH8g0_{sCC+?K(yW7 zimJJI`O2kB*Uz3k|G@_zJ@v%nPd@%+mZZbswjoLk8>aCn&)8BkeKLz`R8SBl%aI({ z{@_7Xf`l|p?!5Dk+i$;JRlWB~l6sdL%OQAX48d`d0%)F3jvqh%&_fTMIdf)rckkGd zHSYreO3(=6Kj4a*=TFVZ?FghwRjq4Bj(>FN`ggzk4i!oEbCzKqIRY9Gh$zmxr^Z5QfVi$*nHSRS8l|31meOogxY}2a z0wOvPCIy5b0ZpQZpjokHf*hU?g$LrIq%l4=XQ3$uW;Qgrn2;AALaDv6CN+tOgCGer zpi(EPo%53d5`vC_-O=WYzuf%b-Oa5lr=NVd_uzeXzvqsmwtv@M$2*;;?|tWQegbE% zz{n|6&l~S*TY6)uh-%nBUPYpf8&@$5l`le=96!4B+_R58aL;Y^WZTRJpBn-q36g?_ zIlLRF0tC>Y>JUQPp4oTt@n97;4MU?@f@DkZY5a`ML|(h523Vn`=O6ykzx=~nAG~wu zWjyw&(we95{NMlQ&9~3V*-KCgnM}PXf=LoBy{LNcB@PZCx3s1x#55&un>-6Tb6w0E z@?n3@^Z#Eurg8qT+H$BqS#&!1SGe!JCiMi$2&9k*Jbd^4-+KPp-~C#5iMOujXWw}D zSAYG(-5Wz2; zpYg`f1y$>?Dnt%`9R+3L!$(^5aL46zRs{v7hoxz>&Au;13Z~`KO#c^um1O~-3ZVuE z2GiS5KKJ#nJo@!7$*GlODaA?n;Ptm(`@0v;zVZfK-hje&)=$)vGUOEoHEfz`O~kC% z<-}CuJk*sXW0;G&8V-k)rOoS+hL6Dj(#)wW3~tOj3iW{Vae8-6k5c;tyM0|qgqyZN z>t~kuKLOdQH+akNGm*S;imVZwbHXli;iicDxmY%h@=CM#p#UNRejwdD-e2uc8he(n)aY6Mr$BfoF zi^81aY_4dA1yb`t;MYC}tImy8eZs1zi|@TIh7*4Fa8haywy1|(Ds8iYwH0y5KnNZT z0|~5l?|kZ!<97{SdGGQaOGnPUb$0xVx1fL{z5e#(8k(fr?W%{vUk+j`PDG5A28zO##;ZkDKZZ zA3$yS82i60%=tj)w#KPiJuiRjbS`oJJK{`p7Zouy^3G-4J+?j+vC4^qa$~mxhP*SG zDMGNu_`3d7Pr;4E^!a{&`vT@30h}^81P^`Gg3K7;jdZ2~q4r)HTYNi%iv$>tUO3I9 ziC>LXEChlOiIEDb$dLjZ1W-Ldjfxlt@&Js$urfxJkSwufo7yxrsYScCw5m=@SCl?S zMT100%EnVmfKZL$nJ7Yvoh*YO>VpJ91wukVgQzY<4Hhu+gror_Fal!$an6@{rKqVj zy)1LPl~GLs6I)N{EA>9qx zuuew8PbAss+=J6@TQbM=ndVQ8V9;At@>#E0U_KT&LIB za})BAdKzBb$S-cKtR5RPR2^8m=hTNw>E`aR*Ix+}4_=}TzX^Ipm^STD7xi=G#Q&i zArLVWF;NH@oGIJPsTf2yQA?$6s@&{#LS`)d_ZEuh=8?s zHZl0=OrZIQAUyC90DxYv`;(vi`10jTU;EnEZn@>;a5zlU%or25>_~r}$8mge4&j2k zWnt&V#hV9-NJ0?R)N-fO8T9)}Vu5&ER-;K?dw=cP_4hycaBFL8{n!a#`)0C7_0VAK z&muC$Os5EUiPkXbs3xk)fiuv7|YB-1zVg} z?W%6CUl#djxVO5za{C>3jE2KAXU;4wErmdUfslz1K+0lLRYkX(#z`S2wAO|Y@*;QM zcNQkEbbpq)&eGZ9nP#zqn7R7qdKmFPMP;tgP+3r)r@*fnyhqzvkzfolQ(d_h+7s9q zz(`WS#ZfNkbUH}%XK8m-PZFE$4R@B7 zSBuh5@||vPiE2UdPe1Y4dvCo_6%$5`<6j{N8&s6pI@YAi>TL9{xzvbApaf`bwp>f{ z)0bX-|IEd2{N5iw{Mb{&yt3IK1nBh#c~J%tRn7A}OYMmhC-%leK!|Os26)1aA0?Jp zh=dK}o)GEesZ%!VT-)3zt7>(9%_fPcB4C&jnN?ghDaJeCCd>c-YEj5ovH z4q%#|x~sF2GL!S_y@&=R0%B{7DEc7n>Pf8#9$#RXQd*{?J_@KB4qprobVjl!5^FMR685z(5q#SsCIxP`Yn;OcJ8F z1O6(6eF(wDGvMrPX&shRoh62!&#?`u`5G9A98sk>0mxTYj@)`I*-NE=uy0w9{KQuIXQA?Z9v9O0m?0;gki`zEeGUpcU7NBSf($>Asr_>G`2 zKEY|&9xh>y#)1(e7-v5ZP02t2LXHxKHkeFA5mp7~45G0nBs2;INCn4M;qKdh|Ns0S zS8qAiS>^hAc>jAp|LLFqjeK|!unUDlMKG-GQN|zxv)2$J-g(}b7v>)?1%*IR#)T4Q z!tTtYapSrXRY^YunlW#wRmH=ULYSeO=0`6hK@e|*+c@NJLL1t%uk@q2&!1_cDq!t!0W z0^&)Ne)#hbU-;2aK6v#FxV{Z_0JM;Bv9kxrlvq>9zY)_Mo=7p>HQEf5$Nt;zf8>kCf&siQEiwA2UzZS*U)x(_yOAwrp(3803bt|b(- zoa|OZINJO2|NdW}c=Vn(E)H)waZB%7`NIGEFW_pi0@>*LcBk8|ob#~|Xr?gR+-jKx z+#qU8<%@VcJ;b#er*VZig1hr`ooHbbdy!DadrOj|%_*Mq%td99gPHyT_k>c*}T9 zGZUca=+z&*u%f1$^|p3*x}DCr$PF!A&-r^k8d_-%cOK7##D_y-MD?KkKHl1-2sae; z78(GG0CQ04`7bAPw2&sqFnYHTV(End01}|h{xZ=e&C_>RM1uebaPyCm98!&_#6p*M zuNpL#Olp$EB#B8386uOWotT{2_;&)504dg$BSlgoKvF~%PzWm6mUeNOFv!q`@Jav$ zw8)&M$OwcU)hDKQ4#+0NB4QO22~vBz+`m4&-nsXY<6nC88j9MaKY{A4jkSy6Sh38t zrz)TaR!kHzAWOW?sIjjz-BY~I7(H!qpbDw!HUU(CqJYR`ECXbgU^>x{GglfLo?6mG zRefu~G^QCoXy#Yrx)^;Ku`B@fG1*fAB|)6eeVrX&q$xX8HN?P3RTN}4q&PciVLajC zwBdwv(hI5@#Ktg<(__ni!^6D!gS_8&KP!SNglYCsd<>dDP%R(`9#MfvDP|;+YTE6V zzDkz{3b}C`Xp4P>1vUpC=!4m5*T&v$kOp@a&;mb z#a^{>W%K@fPUJg?NVThNk|0u96lt2KX*!uq*cesuUJxKL3<~1hWavy{y8S`c=>mWc z0TG!E0D-a9DBs!H9gio@`KWM2aw63dt-siY1M^pdd<+BLQ+n-fY zL_tsyHQE}G5S0)K0ckRs^m@I1zn`XQVlArr+68c4;Q#Ue`k(&ufBrAM!O|vM5Uc#4O9|+8GN1Afn!TB<^;*S*LUT z%Ee-`cl6lt^|h7dr9OaeZfy@o z2K%JaH1B4tEX8TeoGz->EkY?uooyYcQ&xkjEG_q(FWDFgSk`5?-ye_1CT+n6I`6Sl z?|qi0qtRYb_|=sq1P*>uJ9~X&^ThFGCL`XpClzHT5wFsq7*ru*F3_+7B}wY)YBI^I zsv<;VNJJ4;RD*<|5~L|s04XRKHqLwRB_;s^0JF6=Ey~fEv*$OqwmIdeAA7RX?fvKn zKTM3)^|045bzS*lk`f>>W2i%EjWJ20iottz!pw#fOw&m|>10Wfk5*RJR2_j2b!ig) z=GT7rzPoRq>}_?k1cCE>-05`ds?3t5Di5HT5{IHNNa%crd$>nyoin35B>+XAOI#!(oQew^^z>R zwz;vhxAV|L4+1D5*4{fGx>;v?dvk4hu(x^j?O(mHWPKQ2U0XiUu{cP3lcE|<%AyYA z^Ba)$)9wmqgQT-0oK>POcpwYRzyd%B41y3bYzhEE+}xk4puno2$_lVhfZ}IZP{QGk zufrl>nvOac0`CI=VG7>+;L~JPdSk!$@%7!w($XMZ$!s^V9ky1$6>fK{zj`9cx|34L z7`Caes)o_hWKN+1sZtBs)=&07E%d)d_n8n?=PmO4-IzsHAfp1*;=nK(_HBi&WC%bi z)!G3PKmr>j7=aJ2zIEllz4gjlPyhb!JoKe!wFA4O`ql?;|F^gO$M+t6BBXuW3{q{QoIE?j!;y|XXAHhSwrsDV7N2MDp|o?fN7fxX$4 z)%`=h?bRZm&TzT@=658VCOL-#)O`0=~fcN}b-zxwk(|Izs3 z^^Y%HfQ=n+0f0ah9Doo5aNDSFG>FDCwVw$Umlvca;W=&s(Pz~3w(v6X@Y!$IKhCvOUeuj_I@a7xxd{-0)@VzHr@{M{!Z1#Y1fF`+kmg$rG6I zueCX+V|R5a-`h3ZwahLb!D{FDlMmkem8ULFwvOIs}J9O`{NJR7C^8Y;N|c9FkIQ~VVVRIAhs5qtLvI^ zwl44MwJdmJ=a#qm#dZPGs;&7$55t3ehCMa*-}O^XRtv*4v&QiJ(t5+j(hczUviNQE z(=7LR<3Hm;#)~fWB5>X;7StHdA;40nf9o;mrL~8iLbWrd!fPE=l&vMh;5CXbpM5ys z6EXwhx&5$Y6p|Z1RCq8;MDg%Xm<6#+&(liLeX!3l*YFpbQ1h8Yj-ni`SBMRH0ZEBO zNU7A4lvv0(X3qq~#HcvPdJJgTFjB&aL1V39Wt&=!e2pL|sG*r@Wj1k8rM;1wd)P3G zDjl_4-G5&&CFlFoKn*rTg!51jVPNvYw8h2)aE_R6`V<78#R;z*rf zP!==-DA2GD|*Sz()52kg30-*|%iA zPde=$O@(OyJ~U@(fB&c*8`ppfjhxo}f`l-~(P@(e)k~Q3L(lFYng2*5Tmq^ZP-?eY zovwrc#H0YyI`iVo2Y_ytP^@rrODib605_*6!vluj#tE z{gL~|?_8|MxVAjtT(t};4NfGUc?b|tovY(D)W{lcn*d>UCF9G9tyVD!2AM+O;*kH` zSl*OC+1hj4)h!-lN!uDK2#8Wp5kyO;BC#m=AoFvZ`?iwNa0rM@ zWRuh|$0sm^;L1@^RCQGa@0&yQd?l<4zKVr?uQ?Rx4f8s57B~ruVQcZmd*X@5@3`~! zNj`S2?)Uqk*y&`>#Rl*!&S4Iid+6ZJoPIZZ*|Q~&h%~C$B3`^rlWxD)>2(d8EKP0J zsq@M1&hUTwPygG0`_A9K{q6@@mN^e^y!qBM&wk;=@#95N=yc>?=Vrieyx}!zjZKh% zBBHeUjmDVZYGaI7smn@5B{)~rnR95&C%5qb5opj0ElizdMl^p<2*jk`?d|UP;K@Wv ze{3Om1ZBc3NfKjJMZE(75h*8QC4y^c(zM&{rfJH?z}&l!`xt9POyH$1%W^UrkM{N^ zqp>)jFsDiC>KYXoi2xf5uj&mZAW;Fm_u8Cr2|%kdACLA#WbMe&EKB!xcP4pZn2#Pk zs;ZYSUpaN#9e{)sJN8rBn$EXmacY8I7zI&W$XktzS7UQVJs?@^+v?kOzI_mr3gt3(cqQHVe+{(6M6HXF?0985BAW35S-Q zPTz8DJxjQ*Duf6Li9BmUjEp6PSq2^&OUr|9k|fqpbU-&P^>!4ZwHd^{QqnZ5T^(y0 z7j$Lq7*lp_bNG{2-h1(tH}1IS!9Vzu{~_u2H+T2aZr6)Mp|&Z7Vm4`O=hm5~ptA^E z!XS!_Xc&A5qG^)!mzRN10fwW|U@%x(TADtU##jJItO3-W&1>V`%|VBj`h(n+>YZU( zT}^eZKOC1EJELK-TNRsN`p{cX21k;9C zArJ(Kx3dBW5&%r&7e3<{?u`=v6h3Vh`Vd!Pi;|do)-mgAB`_Jo_=8vSuy^tL-A_M$ z>;0#OWnfG9JpbgWyKaB~XTN;&JKql~*0n3Ef^FJcT@e>1d7iMD&P~LU zCZhUi@pz>3fjaX;QnM;zzutau9A^3;#YdiE+JOKH5`wBA_{2z6Y=PEy-r6~R>gXV2 zD>Wb(0yRN7D3S(Mud1K~4N(mnT_gSsDxic48bE7RWJ82RszK&!AjMXLj(!3`5D0~O zFuT3=H%3WC9rQ(fc3#=9ljJ;uFia9}@C2aY@j6`J1x~=1 z8c;-0L3D8D@TsbwV|?2RIcDuI9o(g4bd-Q3>J)|a4f%T!MP&huaR zcYl-|Svoo}7v8!0>UV$q;g4Q|vsa+jgseKhFm|mFKH*vabP0X`0Z$JFg|r1`z?;*; zbdh1Y-ytwpmt(8FepbW>9%iA`#kF`bTZm}7GpOT?@z|-1f7R{&XwcP~N#!fr`x6^yrX6eLNaO&t|-~7tT zskJ=^Yn1N1clo8i`W_6&ChZpE%7G+AMv+4_D}1_9{d2?yeDc-4(KkY*n|k9GoX-0u z=-TP91#%|dfUh|9FyaBC$IT+DK67VU0bKBqnA}yc-aB>sZE1qTyfSHe?eY~@*Ajww zHN?gkk`QeLzsBeVJorJqi3|)>V1YvuS7yu9JTL*l11c?Y%;yP0ty>ZRNKjBKgMzaJ zk~olrssZC{(C0|sK6Bq^||fcH_oqDYpJv1 zP<@~PtVm%RMcPi3wNE+%u~<7wQ-soN>%eSUMCsy@y4BBj*oLI7%`#TuO66tf45Kv)B| zZos%IMmv2~*E-iWi?5AU%i%5q1|mfQWC9{ZQe93{)3HODIAm+hc(ge@cX9XJwdD5W zQr8aBqo;1+e)iFYj~SPfky}Z6il{^lN*vJuB51_>XslJ0x^{K#CwWoVzA9@80UD&voR9maHr%F9FHy^DRWUiZQX=As z6DMxF{WfFF#>U2AFc=I5d%HsbV5T^C^y!=EC+w;SPLvn{kq8u{8B+maWp$<7?V2<( zY;1p^KD_;o8>y<0l{^oD~_TMV@w zlQ*npm&9g(j*S7pEXzujqAa4Zv@FZCt`p|CLXrIA#Xlc5sTm@2lWVt*)*vEiH9A9b?RVqN8dPM;Yz&L7ek> zRp$A4QcUvToN3Vn`J@0qV@xbCB&dkYk0HVV7J2+eL}-i&ArwWC=ea0Yn;2swD|~l% z_u|Ei&bjzcrsLKW&4M9-D(+7pIcS8?8wmTVO;YSH5nH^BMhFi?ja3)I zixITb>6K*_eC542hKM-JvgPHKEX$a=(T$M+5itNLaYMq1X$#Z@{Sgq=&et(}u!R)^ zU=WQ{P3;dBn^YJ?vrcAhvbQ%ZisJb3<8|d;`1uQYUfz29ZL906X{TFMrC}b8Ms-zM zYa(K0I!CJGrwsQQ~+7nMacJ$a9fR9?gYDB;3 zlpVisd_h(F{eHatoO3a%6sA!r4dJJCP$)Gk&!C*})R)cK{mNub5Hau85r?smH?D=UEHoV#-6 z%8?^S*4EaV77C(a6T#&W{H2dCjz@d9ox0`0`|kPZAP}h>!_IR8oM?V(*ZWJ{*ot5{TK&pIyR5 z#F&($ZEF(&0FAXZaR^mayLd|{X_6!fs8e4DUsEcGAR1IwE8C+AhpTsg{h8&HM_&KCpN(F814fli2U$0( zY8R?<5<+UyZnxv>TD56#1%g1cgD&aeBdAl}4^C+=u~#EFk4BK`^!}W>t?l6$fuzB~ z0z@$8R(4p+87*dn46oMTVUc3u5EZ|$Ke9yg38 znTExD_O8|;p5V{4A_k3PN6Y~0a0#{irZMU+8KOm-C_H5UHruB{|PwneO>~+sV!hC@hU(o zska|gY|r@z<(p6Kk}O;cf-ogV>&$T~=?G$UlUS_V!jXmbOp?L`dZ31n$t zs2SkS+wc9xb0?mDXwS&Xk(KjroqzQ&zYlL+fRc=o^G?%DYr)@g#rd-!R^|o_+Qk6( zzjuoJ7mCjU5N|yxpQ1q{;lc5;NLtg@c|8l^RN!_{U_O~s5RyT+bUSxE|I|HSetNU0 zEZe0TzxP)^fD2a$7!W6}A|zF{NR~`3LGah0J5T3f=lR+{)ZZ; zNFw|c8}+BW!kbOsJ?J>r{hWH8dr?k;k{MM64OZ8WA7cXaVl8}h_8d4Llc$&|8l7kn z!EqBk;{z<1LqZ)|C%7KA%;I6%&wNtlK7aSjr#3?`K#06!oO8#|P3_=fa)`>@)D*UJ z02pU|6iAUlv~|FO#z8&xjucTGDUr_!2o)K~vSlD^tYvF$V%b<@L77d8Km=$2S-< zQJGL`b1K?=@3yTk0*joO=JBWKCyG~QCNvOZ3mIAfo_Oz@1w>{^gR@)=ozeN?f7ZS_ zXtF~k%T|>RW!5b8X`0b*AH$!UZ#^yga89Ocyl;)qId*Kx2bUo>v!FH z>WK$Fyt<3p9rTQhoeI>TUd4;37j}XoNot}YZu($GMJ!4c^C5T*0*FWeLA2p_HTShj z2=nKwC_5>}=U&^9c@R|~gs24r1O)(7)^2New{$+sGU;@x@z@nb#DE0veeg_lg8@4P z09V!dc%nhs7!fVYs;YfmyAU*pMgfba=7pQhcYyhrvRU-(Y&)H+4~giOlPA+OnM@|e zaF%6tRaaHj>-E|wv-aOcxYMUKI1lb7LZFC5=rsf%h|yYWlcYcBHF=(F%CbIl_WYmy zx4-$bKmX1LA6#_arb*iE_BOY#z53c4_x$W<-}?6N9zT8z{Is~m>5NBXZ$5xb)G30d z?IcYN-6BG45V6zgxVm<=D~ck^I#pHGb(M5F^N=X`Z3qeifZ#)t*!VcGF#s?gPqw$V zx@ndeV~7!vi9~^kmzI|DJYQQ^WJV@71|s-Rahi5JO|F{?f{_N!%ynHCMKKzUigHrd zxr&dgbed!a5ml?&<#_=}Y^(wTvH}GOOmw3|q%lTBLI@ExnPr)^b}$&c^oy6@eeb>7 z?zr>mXP&+5t~-0(?&ij32%+UAQ4AId5#r))IrN?(XfLKX?AEx88d83tw0o z^a&w&C&AZcSrvsy$kJpgK*u=~BotL+3=t7g1nr9k5DBV?z=TBD>)|aYZ&_JcK6&EU zZKqBgJ9fkvtYWf}HgSkdq!iN;XBu?;vvV%qZ!uPJN~LeIrDkyjiVbat07;aStYg!q zaiwp(_2CbG`U{iw{@{=QvAMlhd+Bx4yr^uNwqBM+fH>^Wtee(gK?IqDXkArG z+#mG%gMqO&1k5Mnk3atS#EBD2OH09fwnnut*ciq zd0U=)@az{}{@klCJyztSot@oVckW!he)HOmJ9jp(?@i$TM5cv8)b+srBcFks$)c>I8r9y}$aV-@>KM%T7B zwtl{G>FV`s*Pi<77mhu2nzUNHZ}s#8t7{LOy8Pth@BH}>?VVja8iJ0LihW-BN}@Oo z6B)tY&vTd&JXB=~*C$VB7O_2IshQloI@J$n2jFP71~F*Fvj+k~CIDnc2U5CxDCO|< zYWE9IKlRAzQ^bWUOG>q7kV=VCf=ZZ#rM|Li6Lb?=jRIkSp20JqFbfl~QcNMzqP`qq zcDxrE#iJu4@QwsCQqH@=<`7ZEB|LcI>`PBS^|e=yKY3=Xq4K;E(cZhaF24TeFMjg6 zyK)_zfbK%!V4QBcwzyMI|+y#pKDV97Xyyb?eY$F8lqaB6qOd&Zyysw(2q z=C%F)NBK-*Vk;=@lk4`62p&x6Xc~UApSi!~jlX7eW%d*$^FEr$3krCGK+0|=RzP_n zFJNmXCnB;4qa@wT6+>@JNRsS<`yTq-Q%`*9~)IDex zBL)S(&H>%LkSu`rc&g_b^fYCC%M9g7@Uu?d;M^GNw5c<_HgTGhJ2U4p_qWaQXF8}Q zZBW`=W=MV60|;q@k*a(FYYS(-{QUi2dLE7q?&jmgB6{Q7Ki>MuTaaT9V>jR-u{?Mb|KU^BLzx0s1!mG*rf!4w%Q}*w z&EfvzkAC9|cq|!i?wvI8)(00Ze(!Z~l!7X%l7WqhKtusXq^$FYn(^UNlTJU216o?} z&Sw^yaX-NQ*CSJSZv1OHW`+A^T=q%7e;q;jo~ST#pznaw%|#H%=#qe$`zwpR!2$~; zq_u~u7cPS1NE;|B6!6ZWASva8vFKjnI`KeUWFTf9hCo57Yw}sN_aWoIc)(40=u0~d zp>B;lE#O6_j{f%U3LydEV-^Gk<3Y`D7j;M-yLyzamRajSIFrP6$dRxU5nqee=n(`I zD*~c2l&VOq2AxGMS|Mqqh_pc!0cpS}HN>du=Ocl4>BW0tMJU3AN~cjpe#iJ>SL;Vk zMvvXMmqO`bS;O{wcQ?*oKB;=>k;9g!2rvvKaj;b5jQU*TVPTrNyWcSczSV7krh1kp zO^wVqlf=c5Rx;au!}^URsykf`;-Ov}4K;%4Y-|?lNl^)^!y9G??qUWO#r52{1awIw zpi*6ja06!%5d!Q*3nX&juq4hqIRu`=MLC*&T00PiVA1iEy&)W&tpJfx4D1kXh_Mj> z&x)mYBAXQBbdY)Ncl;1~5vTOdN9X&K=ac15UIGoM`}lq1pZ)5SYa2Od2{Q>=MXVJd z8PTanEudme(gs6-h$soka#Oy-G zb^DnNBZBc0yM24B8kYS9gK<*CaZwb^94wHYy;g@H_rvREv|-Pa@wh50rIcrI&O6Tl zBn*Uvs0eEpd~iSB`%p-2_N`@truP6q)S!^1o-aTFv1J))oh%Hhq8N?FQ540IDa+cC1V`ObZhsX-4;V-S-g_3OPJ97j6Jc8B@-58nIBZ-4L4{{6S# zc;_RKBu$nE3mvTmVW%wJ-L1P6pu`G<0T7uRrZ>!h9cFR@ri%0IoK3w!)~O|=RFWj) z(MT!1v$K2ZRJW>1=UfA_Xs1*DcKHGDkQyq?I#N5h1tPYnhP!+Dr06d#Mw&o`J%@{5 z5p}y=0Wnb|fR%R&$vUUBiPLn@TaDtR)9EBhq8l1XD?o?e8e^g;isP8YS7pu&;-N)< zi)Z$d7bPMoW$MvNXj!I*pDeuhjlc$tRyUd*)1%CY80e%n=ZQ zy{*K#EK9?eAw|F;WtNcZcv2z)KqgQD{wPFBDWj4!I(BUJ*=L_xU0LaNlBMN7Nad^* z@mf(6pa5+ftjPz$*{Pt=)^*MWcR4c)h&D<9PzVLFhA9aOm?+Mx%2%**{1}?80bDjW#U4s$9~E z2a8MHZqGZfOfnpfwzjrkeBt>tO_^B*<3f1jh<3N`-n@PVydCtjeyY&f9=W9?e(2=c z>Y4jD@9tcNYBNAtZ|C>3L3G7_oAMLsN}G>(+1N`M>NH~#c{H$J@h#H%kp z_th^fK9FsX0ZtB{{iEMoJA23FJ2>AHDa<$A9v}jrT6{%{yQsNK$YNV3K~f$_v~d*cc8f z9{&FEH+S6NePK{|q`9oVd0B+Oi-0&_7n;Kukr)s6uRZ&}{=?^g_sdZXyJLU-SMPrR zU;SzE=7(SrDu#*`s3gglc~Vvqo4@BGN$ah~BNc0z z7q5<#o@cB4gF4z#uZ&=wQ72Olax?lX6zs+#j+qs*LeMat>_W%TLuVd(>hULEe*Vnk zkEDYx1AO@U`yapa-upj)gFm?nTSLelsK~Ph&utD>fbRs?Gn06!p`Q&zSOi(-!`tw{ zR74GMZ&LcCz2pw2ryo3z%RQPL;hr@qMqq42Zk!Ra=V9Kn_Xi%<8#weW?}{N&Eh-hsR0j$~X^ z;#g&|w-$aQ>(k$51jn=yjxz*}wof4QOT$sXXguPaA)o5l+CTSKV+nw!>E=`CX@`@;j*@PU3R&NHL-dB#!ZywdvogfB*?rHPG52SC1jc|G!tIYP z7q_=kb%McH6tY>F2$2O)kb?tgN||V62|?u#P0Hj0BdfKBi*C+d*BIyO@MoG;rb8Q2 zWTtD=OnZn{y49;&ri2%ug^UkV9UDjLKE#Unse)cpc zFm=!)0*DV*!Z6TKQU@@`w#il@JXn1&Sc5Cgs+ht;t@F-36^o zmSy>55=P)>@<{i$Mr&t1lJm|w=PHLt_46Wy7-S1-jPtlZ5a6?}qFgsCY6Qm|ORqKt z2xB8EbLbra5RvyjN#ahYJ06dy?S4B<^&3gT>{f(N(p6=V6e2MLk|L$WSwxZ}^nu1sP9 zPdznQT>RCq-hAV&x88m4{oCssgQaC>t#?)#Gv$&au=jbMcapg2E%K+1v(7su=IqQh zXMBDb9i~wmEqy?yZ7}vO%hD`MCu8fq(@JyAn=!NP=0?l$-dhA$Rn=&er>S}Du?Jsx z{+UxJ*NDpL28ALG{~hQ1}0KL77sTR5t0xH zT6P*mD;x~^XCAn3Fc=j1s48=%8Nq6WM4FiiK?s?Bcw+4^Uwb2Kt;1CsK866|AW{%P zb56uY01$+#v?fZ^{*s9o-uw9KpMC3x8(ZTqef9S~_tF=b%rLJ^lA$)_#6*}ErNB`! zAqP~7dU+ww!U9U0PPf|+nSb^hYch_BxnU%a+`_2%AoIhM)DRznk!L|vsj zq>xE5kQd>w@SyEOOqD&EKY>1Cq<+ci4W5F)L+y5 zE02T-UJ#vPNC{4#I`+lq9{k43r=C4Wec0NV*d1Gq-L)V8{O!N|VfFDXC`845LbBru ziV{$9rV*&Y8p ziUeHYKl;gQKl|_hCwcEO=pI@D2gndbkuj#Q)-e+y;0%;U8iDd&wG+WZRu(xtp#@-u zr7QDE&vH0P0cRnjIH!FaYJQ}7H<`$suV-#-dpheXhXkG+5p)*I5D@=4cL^H=}4spo!IUcHy_8s6b)b-z1p*0O!qqe8g5fj<28hh~SvBn9m?k z4lFb{6Ba3-idDIn_Q<|d?_6uD`Hw>9d1l@seBcNQ4c%uRd-O{$ao3dI^_V_*{Z|)W zf5SLb&J(GK6e*>>fMwFEvKEa-JTNKz8^NGO_VbNcMBzO{Ll3=y2@ek!trwX?8FPoT zbUzakL_~fQM}vG?PvYQlRsdLG!Il>Ls}G#U2yE&5TJ7E0l&u{jNCKW&5w(lVy&j0= zfX%Ck%R#_7k8}8iMfOS0_2Nv32TCIr@H8Fz!P$Qm%uY1|cY0Tm%-Eu3{~(LNl%B?I z=^^_BM}}D>+?0QhDCm9FhUU#4S5Sk@>T&g!!1;Kjgb0XOp%D7O3?rKC5Rp<^X`@Y~ zjZxY}fQj=1la;ZxAQ-`>W3X~#JghYVdB+2thx8HHZ1=whic>_BgxfQF4Evqcj z9qjejUQPc={Q1q^dQo2Agv#&Qa#$AQ@x%`cy_YNNb6a@v+&qNYJ1;DqPSWaD&bc78 zp@*JHiADSj*ja>`RWM<-(O0#nOgu25U{g2+iz6cl5o!ZKS{Y|2J6n7CsPML@NX2Pl zqNuE@kXhQkM+is6@CqO_S{n-zC?XJQ4znSYLjr{Z@ty{vq&8{R3_htTpN)}OW+He) z>j03n)_Z$f1N( zy!q}&fA-Dq{`t4Qf9cXS0G;%gy2){)Vnk$iBHH_6dwX|dWAm}c&Iw^{!ZD_xo3`UKNqDD5|pb-fQ+yZ*+gVFwi*Pq@NW7RCt%Y zM*v$@TX*k{hQlmPL;9d|4gn~R0=^jlCXesn(zba#Nq*YmDNt8@RqcYFgS>Q0gA^}p&5^S+5SmKT{s8KT{69nk+9hv#Ug$ql| z%fIux-#B*SL{ZtDy}hC+_J+IGTBUUhNJ0STY*FL@3;XXj;ZM14_gy)71RJ`cHLOb-`*~fFpen89y}b*ceDcoQ zZ>=mXz4X#cD=W*!=*f7TBq@7GN)w=CFYF_u(5*8=O^o1lc%5m!dHLyAL^t7sEEKZ_|nBmrXv`QMoApC17Wyt26isLq+WiN*3L7M z>UaBt#U&HP0I0MPf$i-bB6aH2>3VGu@a)7h#hUZs#S0&A+_}+BBSXA(=f>X6&E1#P0sr&&p|xoVwU=L zbF%%-AN=to9}!KI|Z?>cE-w+=9+>$ zo#3C7BkD81w6r{&U6sOE(-EKvX~QAwL2VQ;jM{?;SQOL(=mZMKBXm0e`I!gKJoC)C z#icB-Hre>7lMeG;%>bV2gMRJJzyo#~IlvJQKMPv`LTw0{=bqJ!{^TvQShNsn7(}qp zKlR1uzWTrUr+ziwswy*7C;Hj&qxEn7fB$&nN58;I!CsNYJ@#CTMnIG;ECS$UIMh)b z#5)2MZiTOV9(n%z$7h7$eo7@gdyrTg^m7EV*EX9Q?GU~N4x zXWeGh&WhA60~!y@cS5N#(6j0=;C!1~+ByGeEOHBsYQ82OIuuvZlub9Kmk<=}!4MFE zCWr_!1HBYF<}3g8e{u4m`&Un|Ie^O>TbDoh_?;iTR{Zo=urUO$6-J5#SVRM80$;mz zm^};Eq<2L~C>4vIN$6;$>nt>V5zq*ZIIAGKOAuzIK?oG$}Zo#Z9BA=QcI2E9! zscQv}_J1DnWois(J*(9oh54|szS*|wr!@>_FKm%{MNM!Y*;-~r?Ajb9`>fI-Us0y* zjqnMcfYHI);salP;gK)D@AVc z{Iv12{nU6bSnye{iy%1RTpV2Gedm)ZwH(!2*c~Ejy{lupn~?F2UN7$C+OF9P_eeE zwQUOJW}Wh8j0u=Y0jU*12qFbdL-2%Ks-Wpd;ndCI0nj)3)m*Qu2teQ!ic+kNKvKE2 zWXVV9G?@k5)ohP-7N0z3@6$U8+^tIR@s(?z+}_zZyLNoz+OCShbNF(Bf?3FH{H=SUgaXj*cBOU%2X98bTO~;p%&p;1YMmA&Ns_5`n5b4Q ziKC8TI%Ee3Zq0oHG>8(65p-ilt|e#zFwxu|U;jA*ngHmS{$cn8QuD~N=5$oLxkFUD z7JjRA zE32JFtDIkdVD<4>w2u@;QDmYH(V~YKV0YpyJLjCW);Z@Y>zrfn_xARzwM9{kMx*g? zG#-zOqVSd7+SmjiPL@&+2#PREfLpXDXKqp0;Zzk^TP!Gb@D6GsrJT`1I%nA}%86ww zh=?YNl`-LXLr8>#NLYuj?B~1@pjNs~Dy!H*9Zn?887YjptFUVPru(YQ#38ilEiz*b zrf|uWMUiznN-3?iF=l;zU2C(j(61^xla5&naCnbN0~pfbJR^WHQLoeMbu#u`7S?+I z&f6dSn}7SKfA$yOyLER@M}xT2_deF9UltBfK#bBUxQYdtWh(182g2!jVdjao)jCe$ zA=3|DHSH=vn(5pM$Bxjwblx=c!qG7{_Rd6oatHC6o2-P5WRDw;a-{N3(L#E zz7C-y(xeoT)Uu{%t)nD~l5BN#RT>KaK2iABINX^@sWizV9d*+zACE?R!+bKaRVe~Z zY#ktCo1yn95AQv*3%5oXXh6hc$Bymp?#4;-xz9a&cYEjR_3Iyg^zrqp*Sz;o!w0x= z;<2^WoOP&`iE$QNd1x}>OakjZ9{wTFxF$qkh96Bt3;o{G(%|;ZJ0n*jc%>C7^v+I? zlgI!*2+*FZ!Jv2Uk+Y9KcJBWBPp_^ngHh#Z7tqhl>VOD0gL%{bHO$PdIp>{o%%W6) zAlG6{9l6?w!!=*1Ht}F#U?Pnu z;w5whYHklVfePGI%FX?$wbGhE*xJh4ioMmEI-S`2JY07O!YpxQ6dCWBjxaQY;K8<~ z=e_qn#B2tqKhJB#G+10VN&m+B-d}$E$3J`Hz0bY;rQiAef81SMb1Ye>ljlX{sx(R} z>#TLXZr4>6?yC-Eej;pszF(WTw7F8Mx3JJ#SO^_HrFB)g;c$dV3k!qr@11q7eyURB z<6Aebj&`@#PA>I32~5UXV7$9K-rF{YdcDr#(m;uGwun`-vXCwgj-Neq-(%;7Z=b*T zvtPVDx;VT$z6stkN;q*bvWW45+Cu{4L%^>2B2)%*B2aMZAWM11*ZQlZBmFN(` zJb#Q52%(oO1VGlYwca_$S|e~0YlY7F3dABZq8G*{;a#R_6gZPIb}ymaztsmL>OsE% z@RKokk8z|C6BEaA1jxHPBR~&>j(l?SXaD#A;iI3w`QktL`iU2x*xs4MWFGtCQ)eH0 zETmlhz zr;O>CNQt=0dgl=cj4B8{Fn9(QAW|qsgq%%OB@cKL7{*he04Y|l@LKIb%6#$haf>4wd0ALik;K&dRB3MHQ z6c56OsD^mBqw45ujgH!`ME(X-=|4-M{^=WMtBAE5_H;b(xB1A@bZ>VT5YtYFK`LuQ z7ObE^l5{~8dFku~*VZ2WgI6E@`pb0xa&Ix&xxVwkAAk4qw|@k8cA`EKJQUbg9w9jwOUCS478$Db3AW-k^(Fd z5C%VNh&E`xqj---E2TU$SPR~RA^;*FaU3=_r4%Wu8Sm4iVk9C4Vdi!KD(r#`;+fgA z2&*WXnv#ME2+PL~&)@;S(z zJeNs%`{K3h?|f9Q@4#RMBq0yvL_B+876u@Q4R~V4Iyy*X$`y+UIFJ|x03gz!6eGII zfoJv}>c5(nyHhDy3D@K>h)kkG*@F9ZlJKP-Z6F7A6r?tBvAw{IXI$xGbk%`TuoOp3G=rB$m{OU_D|0lmU zK@Yjg3jWnMe{}16zl2>EJFJuvgMj22xnfl#^?xht*l+B#J$!GTcYC$PYP|-|iZ?Uo z{n?VH@z)^0e&Aj>N?}uWa!Aw---vKF*c4@spCPl--aM=QGo~k5_Oi1+a z87waD=HpUHy1cYA-W3IF&p-90fBFY+Zd_gIom^1x!|%N|dFLYl1uQ~XR+SRej43E4 zES%*?(Sg*s{j&M2CFf{~Jfz0A`AgJHO5fA_9-O)NbVba5{n>x1l{!QYMsVKar%5FH zPXb2?zAb`arhB&EBP_qJt-t_ruj972F^f(1;UdSM)4 zf}1m<&Y0QT>mTRm!MIgM<{K@FY9VvFsB#!+;Z{wQW+?+ic3v8eEdlDrjpn@%N2Z23 zTuU%2bWx$k1f!w=5-7?N2y6UOXljCxa4KW~!;=bqOhEx8MlWp{j3MvdGfU}9t+du! z0GND81R;fmw?2@05p|?UDMe%zY%5=p_^^ux4>+J8nkeSLuLL9|%AnGs5l^hFZKdhT zsZ+;KpUjD$eCAO)q3#&Cu>jl3jaAj_$IDMW@XYB42baeyl0;cnFApp@wew)ra8ti1HKC|s32}jGv!kU1Ls1XThWJ^12HQM zO-nF`(OTx|EK@u$MKjtq8$Qh)5)@!X~F41rvs5PN!00ddcEF4 zzu)Wjy4`LZMct)^Afd)lWVDH+$QT0=x)>@3XI)j6MP5{8=_s*=6$b~@hsvMAWIQu?j8es%iv=|>-ZYh1mp3z-&-njYqas+%Nz(2+f1{- zkPXi*Acm7HbKWa$LVGEWlfhtdyf*}Zjg5`dr%#VY!^vdaGZ87}oYUI;hfv)Z^s3gS*Mvo zWamhkLz}uXgCuIQJOPOU{#hB5Au0rRzATC$<1EXpwMAJy^2oXS@4xTk3!iLnZ%@XfPPYRh${1VZCW)Q3lkvDHiY!Y5(^+XH z49vde@Hg7=TsJK&$aSMk7=+u-X$U-T`$AYZCbg}Sm zS=v>kMKQDY&a${DQj1Id)zziN#on{eJlXB03xjT!C0Zf#gbS~zABEdBole zvx;J|&RK7isq1j1y!U&1Lj;v1>3EzAs5I+N@=^q&CUDVg^5TU$Dv0eQV(QAQ53qJ zNt#w=K}sov@o*f)S(J2ogVl@I*8l8VKYZuCi%&lD;_v>!KR$W?gN_lDOo|eaqB!xM z1wbouRdsY1#F5fO^>_+16Dm^L3pnO1N&5?nWFi1$&(2$=qHlfc+h6f3S4#P*3I+pya~Q~?BTN!^0=5}u~A|FtnBXAUXd4_PL^~!afhRhnUs5j-m*aQ z@WWsE(iflolW+axFJJ%Y>c+;z78(I!hIlbDx-6aZ#0Y>>1(^clA5~l09z2&GCgJ)( z)^B!kY%B(?#|mf5DI#h*IrZ_FKq=7?rkM4{<({i1^>~V~F-j(5h2GiRyArvwmdF_P zmb3;DK|uilh8n4aKyv6oepV0U*^zg_zmdB5(0K?TaO@x)6WJ)1U@y#l&Pq@#(FQ6H z)`7yEAHDPKU%q$lpZ)HWU;o0%$4}hevAaBe>Ysh_{+FNm$shma&7Zvs?_aTbmG+j! z`r+;##qnTqshE^SQ7Tl%MB>BjBRS`lLJ{Bq-NCw(7BZ1UwR$wR>YBGsD`S)*0b~d` zK~e~swntmp>O#4D6WnBQ-@@qjO{~VD+lhoady zs8%wtwrVE~rWOrg& zvQk_R;p%AW^V-|nBdNvZa8VHv5HJc#ebivj?UY}q)0r#9n;GF^JZ@dGVG{}w^Kq8e z*+)(3&CD#KBBKxpNt$H{LxhB{9Zx))1_hX;F{ESbCWEiP@c7q0k69EUnF`+j!Ot%JLCBl!S~a0E(w6yk0#ZG1jQ1LRTtl{I+o&Q* z5|_3VjS#7g$*#Cb^5BE#UVe6)?b@jm8R+Z3x^U;s4`8=2pb&u(2tawz42l7$4klIb zw=ZN;K6|$VX0ac56b)mJF@41A&vH*QbuQOp<=SNZ{6=u|{WQ1fP|jRlDRa}`#2J|Zqb5r|tZ}WRf{rzj(kf(q z02)B$zl14W4k~{jJ%LQOR2C#19JnDs$cp#Qc_${08h3&SA*!0BkdGz~5EUUxn`j3J zDouRYFvZo2xkgrslDGs2DyI*VG=!vpg|PM*iu2C8oS7X%p@ovT*Y7PZE-tUEE-$b2 z7Z#Hw&9aV&DAHNhSy&kKdTAVkLeK=3p)^nsTv;FI9ho@Ci+Thl7|N7U0+XHX;hr`i znK9QkuL#Ig7;eh2-JLyw+N&FsR~8n5A}_<|xKRB{{c7*-?X_Mwo=FxQ4@d30&oi_C zc~Q*%N$o-qAkR4SCk-~VP4sGArZqu24tnGPq(B=8HosU=IdKRihdHL(vrsF43rLhi zb*{|RmM)DCM;LgfaAguqC?%m;6)YT+4ip@C9&Vo6Qh?Y{ zUC0~6eu9baq*<@qUFi3Foo*-1B4ZX87n3ONE_6;sC$%xg7}5$!Ew6Qi!Iw}?xEvQ{ zUN~!o?PRjWe)pZXzW?w4DtY$-3M1wsv*uqb)!^fryBMTip{0$hT?L* z)ZYh%{Eg;@>t^gUjG`f;!8A#C>QD}Sh%iLoa$VynJ%9duRaM!17r?sR&8-hVy71onAOD+w_sx$!xwbVb5V~1! zRYx5ab%-)RzS4fuW*8)}Mg2 z`FOOuvsD&3B4|aLh?|D`zI>B7jkg@uJUGNa*MKFOQQ zfZDaRE-0iK?2=Yorg>AdG+{yV>jRNR=6~dQ4|RPu^>J(<(x*=!fA!Uu7W(}wm#=MZ zZjVO8Buo1J-m$fnW5-t4jx8@QFD@?iI$6SOjYg#rz_r%b zS%3(gb7_)|#<__SrSxbx?)CaLaJYpIHe)vPubfN~dQ*p*v^c1S?x6t!EJ(nMj%S^}F8tZl@QT zRYb(0eZn5aZ*8uRcDDz;EQ^#c$A*9z5L9@zysygK3Fl?yVL0h5RB@+IQfzPDU@p!* zc=o^gFaALZ^!i(uFI?LiZ9}pEM#pGkqm_p?2dN(U#wl&{3mt|RHaikSSnvzJD1uJAlVl<&9MM9qY+t_;3cw&(&maGrzqtI~M=$-s*B^Q1`FMG8 zw_;rGy!y}o@RR#b-T2k}>u-LzcY6l}(qoIRv|G2XcUO*)4+t}G&I^b6DVW7{2(*cS z?QFKp%;X3CBopVIt=M^uN-O1@os^YC3{f@mJIRA<&%bc4qsOm(^PS<9JFkA>+=Hi= zda+Ptft^?vg^78V|IyCo)|EHjgykMthn4lE8x|I< z4~b;pJg^{wF#x5JclV(3K*}puLxtx~m~&_a{9lU3^7klcm^qn-+&L3PsFZciISU{{ z8p7b7EdiZ6b^PpCUwr2Gz8IZZ&V{eN_3?Y(`RUbfy$08|GKmP(#5yJdAVEC5pl>%4 znLRtmJ!Yct!5bMNV1$5()Xa1$ia1+|XSVE%+_yazw4BTpsiW;qg9 z0TCox8H0$ds)Q^BtswZRl?MSJ16pxeg4Pg4pmjI|DW%w`!j2)0p|jK(3>Fs` z2aAjSem_Z*ZntZqsMG0md%bSA8^^IyN&#?QBwD+&+S=G0?(B|=oxAKNuDW>V#>MrU z!(z1j_`@%J?MufWJgo?fhxIo;S^wbT<##{WzIqFWd5XH<>Gt})sGI2|&U)Q`zrWD! zb>p;S;!Zc4SgZS;#bo(pCySFrX-!&%qd{;yS!=7RDvH9Em2*{Ij6&UARaG&`N2Aer zJTA-f^5x6kxw0&6S=A145oW?J5fcQoS|em2fVCDxAUKP|f1X1V9zYPMeW_r4f=ra*(9>=N5TIgFH6X5@%OgNU+}qv>)^nwlQA%l}ltQ9W zo`>k8CW%KvO`WJ{nv~i+1aVT8JcG9b0D@rLmZ_N8?q^h6xtdZj6xl+8;%uz62zwMt zqXIo71Sd{C`-fjU{oErJ09CU2;ib3#^sn4|pJ*SUpdgY!Rs1I}ist zEhAgaF5LSlbLU@dd+{yDqb7#`Ce@>ABW9-OGZ6qC>2Xy^8bNCOcniAG>CZj(=oent zvc>7sz4CJL(RY5j{i_c!_pvh0Rzegwe2RetT!_=8zw;TrM`g0YVTmpRpo!!N$J5{7 z+|psdr$d!u3Z1&fB&!H!d80M_CR~FB7T-w8eWt%&4L`w zB0-66nyj_wF6^G+69B;&BU4VFv>5=}OCV%47T5Z(;yU9l88w$6L=$vDsO$cl~2ehV@zaBtaW4(qm9vexVMLb=<1G4Sf+?T#yKO80LB7$ zWIDoC6h%9}Ry#Tgou)k2l*GNH)62U3tk=sroh*wZb?VfzuupXCa$8QSYFyp9zK%?a z5X5PtmDcRMKlF9MInfto#*Bf}VB_X0nPsHibK**{p4HJ_P@A9|W>_u3x+P=3DRn)%Sk%lh=Ov&U+tMpar_!?rPFmCY`dVvIIfp#aR~ACW_NyZ1#4? z@4x@i-Mc#{vlEEqoU5wR`6?u^h1tw~b2;ts{ys4m5tW`Y)0Ng*YZFClYik!TUVQf1 zXAv@-wWs&YIWZf|dmMvHx5ux{v**7Vy z00ewAUak?KSt$y3*B~n!W307+q?Ah2WOaEZP1B8yjq!LCnx{zEwhFxW)>a}3;#?NS zK4de_Ca@jJb36E7L%eOiTENo;gG7;!FrR1@J@oJc$JSPF-&)_^-m%u2*z|gx<>ke} zVn0bDAOY|o4nih9IPaYIju4d6U=T__(a1`rB{6GaT;j?e6sZ{l}kt;@F8()>>yv?=6W?YkfiV-l0+~f=ESP?CcIt z-+y204M&8A^sgGO^dYBsA`0zIYI#s1&0^DvX^3rRF*a#-GmRFUAMBWY=+Kv7?>qpa ziV_v4CFARFy!-ku-yN6o)h~bkrI%kFEU#6}Gf>7FojHF}6`U(O!t+PpU{7j@xfjo( zOx#~w>Mt$>p|h5>3bT<{u3Q1Zm6a7D<(*TC*n2|o&MI-2FJ0K#+BkV^xtB(^7{!V_ zvmn(TIAONz*%g(bvPC5WAj^F5llNDK+Yg_+|CJXXb*L7Xe)>QE`46vc?o9SfjD|py z=`pH0C@{o>*03mm25B<08-n%FSD>&sFKLv-N#=H7 zl9y#=0V1U+j&znpyWtAqmz$3CM@9o%W6E{Szo_-Mzfq|H~iq zh3jy4XjH5yw!txo0Kvg3vfpLIbH+E6Id1CTKmNw5{dwZaz5m#wFMsZ--+lGOC5b@9ee{~!Mv?v7w*2wnix1aN|PZYu(fK^>@%gI>Tf z2v8^G_AMiX5X??^?8J#Ujyt_>zu)io`@K##NfKp{?cvMlqWD9h4X>%G5u>!!7KlIP>`csw3YCKF$lU>#I0SVo&xXk$Aq z%Q+k}sCtAv5ct9;vj=JX=51mUQfqQ2A#DIznAv(KRn>YNmX?;7Sv-61EAPZx4sooC zkY_B~bsdMn#e>blRaYq}tVx=tMU&?o(BFbV@oaJ6(h?eL5-QY@B;oTpPUD^7t_4Z^ zi!j-R#qLZ0_;*e}_e7~7wXl2j&Rc)>UH8rf5F?g7)&5@My0ydrLikyz5q~R|rWqsi zK9KQIt#&z#;czIV8Rr}B*q%RR7Hc505hC>&if~FRK4{eV4CqX3V)^RkL-Vl*vBGF* zf*xXm#~wWU(lect3wPZ_O4xY+;>};4hmF0y%0#3RraA@+1c2M%IQi>Jt_Z({UN6FL z%%I%g2AJnEV-vrA@ZpcPImo>rMD~%L4)H!7(BkjpGgDpxFxi!`))gEK4a9ldUMacOa|xVW&ekfw>m zCOfg}G{te0Wm&h=i6Ry5KBek+q9|PGdJgkqH1g|LZ*WnLMx(u*-JR{7y`9~BGd~K9vUD8 zFoj68g7Rc{gblnJ5KJx9qH|@f)*&JhgxV0Xw*Afq6ax@|X*w!0H%T*-COR{8;Soy_ z-*AZ9xe5ey*3QmcG^?2+Q4Nz}u9RBtE+2ONv#lmLFY5&Pm;*Aa4jF7c_R@=@)*(B) zJo=zjNnnv-8!v+lj8pxqHMkv#`U$1`6hS;wu;DDR7D2!Oq>$tvU*G=d>amATZZe|4 z@%v7x6Kj4mHw>9m+5lpjn-WlS|W^fwjwsdblqzTE)Z)iB5AEr z;P%a1Ws#peaUzMMojmV$yG2pd`{5A_VxH#$k|arGtI=qLi2a4$%9#fzTf3L8T>bdM z#W&tM|Mt5dzWL4vAAE8JAc^8Z(piq;9%I5{0F+}C6h`zQ+Qfn>s@VDB+SQvEKe=*Z zZ4H(3-WNqNnT)iKfWj+?vmnHS^XBtlH`VZrL~El>6h+aoW5<5}^Pd+*VdAJLimEEf zXdT6W6U+bKz!7WaN2?huq#6>CcU4(9TkY-b7I|JwCd@vHG>9t8qO2@7$+7#crSW)N zm1PtsN-1HUj3>5o?0r=hhmygB8K;A%Qe*CfH#axJE`b%xfN2qr07)tQ)1*jgs%fm^*-gB) zo-G0caErxOs($k5{ERibQJ>p<9`=9=3BdM!b_i3bXG5?yfB+$>wc{s}Gz+4+HW4WW zAeFVBd~z|%I;*Q|0m`AYM&wXNmgD@whaU`gcbkgr&7*B)&Zh_vHHNN_y6+8!#6%G@9dR% z0a^ihua#2bEdziG7*0GaUk3wt8*y$dBfKTf5k^S_3afkqWdWTG))ohk-}mH8Pd)kY z{STj7y!_UWfApO{h0)fRe(#0npL}q+Z=f22&+Bi%2p%-twY#sq^S$(cT0Q%2vU+0e z?8BXv)wTPUcE(mMc9!m2S$XB@XOzmT^6t*A-P+xH^Ss;KzPrA;y}32q*{g;V*c(An z!JrS$I~h80pa7C6dg$2w&p!6@fBpwUv|D7CqByf~{oTuNeCr1{Uw;#}#xSWMPDQXN zEfA5?ypOKX;;(1(Ls3FrHq@}Z!;ejLkE76hXlT-Zd_L=0aOCX#NLZkqbAX_-)PsPn zfCMxzK5_P$Z@l`@3r`O28?4{0KKSX+-}}+e^4H#h-3q`6>7jEKgHdS~cShxe4`xBN z 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ν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 U?SpMAcmMzZ07*qoM6N<$f_z>9;s5{u literal 0 HcmV?d00001 diff --git a/Run.HC b/Run.HC new file mode 100644 index 0000000..76f820e --- /dev/null +++ b/Run.HC @@ -0,0 +1,3 @@ +// Compile System in Adam task +Adam("Cd(\"M:/System/\");\n"); +AdamFile("M:/System/MakeSystem"); diff --git a/Settings/SystemMenu.json b/Settings/SystemMenu.json new file mode 100644 index 0000000..0c42efb --- /dev/null +++ b/Settings/SystemMenu.json @@ -0,0 +1,40 @@ +{ + "items": [ + { + "name": "Terminal", + "path": "M:/Applications/OS/Terminal.app", + "icon": "M:/Applications/OS/Terminal.app/Icon.png" + }, + { + "name": "Accessories", + "items": [ + { + "name": "Calculator", + "path": "M:/Applications/Accessories/Calculator.app", + "icon": "M:/Applications/Accessories/Calculator.app/window_icon_16x16.png" + }, + { + "name": "TestApplication", + "path": "M:/Applications/TestApplication.app" + } + ], + "icon": "M:/Applications/Accessories/Icon.png" + }, + { + "name": "System", + "items": [ + { + "name": "TempleOS Window Manager", + "path": "M:/Applications/OS/TempleOS.app", + "icon": "M:/Applications/OS/TempleOS.app/window_icon_16x16.png" + } + ], + "icon": "M:/Applications/OS/Icon.png" + }, + { + "name": "Shut Down", + "path": "M:/Applications/OS/ShutDown.app", + "icon": "M:/Applications/OS/ShutDown.app/Icon.png" + } + ] +} \ No newline at end of file diff --git a/System/Api/Dns.HC b/System/Api/Dns.HC new file mode 100644 index 0000000..1955f91 --- /dev/null +++ b/System/Api/Dns.HC @@ -0,0 +1,28 @@ +#define DNS_REQUEST_PTR 0x300010 + +MemSet(DNS_REQUEST_PTR, NULL, sizeof(U64)); + +class DnsRequest { + U64 host; + U64 pointer_to_u32; +}; + +U32 @dns_query(U8* host) +{ + U32 res = 0; + if (!host) + return U32_MAX; + if (!StrICmp("catbox.moe", host)) { + return 0x0a1400fe; + } + DnsRequest* request = CAlloc(sizeof(DnsRequest), Fs->code_heap); + request->host = StrNew(host, erythros_mem_task); + request->pointer_to_u32 = &res; + U64* request_ptr = DNS_REQUEST_PTR; + while (*request_ptr) + Sleep(1); + LXchgU32(request_ptr, request); + while (!res) + Sleep(1); + return res; +} diff --git a/System/Api/Icmp.HC b/System/Api/Icmp.HC new file mode 100644 index 0000000..e52040c --- /dev/null +++ b/System/Api/Icmp.HC @@ -0,0 +1,32 @@ +#define ICMP_REQUEST_PTR 0x300020 + +MemSet(ICMP_REQUEST_PTR, NULL, sizeof(U64)); + +class IcmpRequest { + U64 addr; + U64 iden; + U64 seq; + U64 pointer_to_u32; +}; + +U32 @icmp_echo_request(U32 addr, U16 iden, U16 seq, IcmpRequest* request, I64 count) +{ + U32 res = 0; // low 16 = ttl, hi 16 = payload size + request->addr = addr; + request->iden = iden; + request->seq = seq; + request->pointer_to_u32 = &res; + I64 start_jiffies = cnts.jiffies; + U64* request_ptr = ICMP_REQUEST_PTR; + if (!count) + *request_ptr = NULL; + while (*request_ptr) { + if (!(cnts.jiffies < start_jiffies + 1000)) + return res; + Sleep(1); + } + LXchgU32(request_ptr, request); + while (!res && cnts.jiffies < start_jiffies + 1000) + Sleep(1); + return res; +} \ No newline at end of file diff --git a/System/Api/Ipv4.HC b/System/Api/Ipv4.HC new file mode 100644 index 0000000..105ebe3 --- /dev/null +++ b/System/Api/Ipv4.HC @@ -0,0 +1,9 @@ +U32 @ipv4_address(I64 o3, I64 o2, I64 o1, I64 o0) +{ + U32 addr = NULL; + addr.u8[3] = o3; + addr.u8[2] = o2; + addr.u8[1] = o1; + addr.u8[0] = o0; + return addr; +} diff --git a/System/Api/MD5.HC b/System/Api/MD5.HC new file mode 100644 index 0000000..b827b4d --- /dev/null +++ b/System/Api/MD5.HC @@ -0,0 +1,144 @@ +/* + * Simple MD5 implementation + * + * https://gist.github.com/creationix/4710780 + */ + +U32 md5_r[64] = { 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, + 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, + 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, + 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21 }; +// Use binary integer part of the sines of integers (in radians) as constants// +// Initialize variables: +U32 md5_k[64] = { + 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee, 0xf57c0faf, 0x4787c62a, + 0xa8304613, 0xfd469501, 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be, + 0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821, 0xf61e2562, 0xc040b340, + 0x265e5a51, 0xe9b6c7aa, 0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8, + 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed, 0xa9e3e905, 0xfcefa3f8, + 0x676f02d9, 0x8d2a4c8a, 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c, + 0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70, 0x289b7ec6, 0xeaa127fa, + 0xd4ef3085, 0x04881d05, 0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665, + 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039, 0x655b59c3, 0x8f0ccc92, + 0xffeff47d, 0x85845dd1, 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1, + 0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391 +}; + +// leftrotate function +U32 LEFTROTATE(U32 x, U32 c) { return (((x) << (c)) | ((x) >> (32 - (c)))); } + +U0 md5(U8* initial_msg, U32 initial_len, U32* md5_h) +{ + + // These vars will contain the hash + U32 md5_h0, md5_h1, md5_h2, md5_h3; + + // Message (to prepare) + U8* msg = NULL; + + // Note: All variables are unsigned 32 bit and wrap modulo 2^32 when + // calculating + + // r specifies the per-round shift amounts + + md5_h0 = 0x67452301; + md5_h1 = 0xefcdab89; + md5_h2 = 0x98badcfe; + md5_h3 = 0x10325476; + + // Pre-processing: adding a single 1 bit + // append "1" bit to message + /* Notice: the input bytes are considered as bits strings, + where the first bit is the most significant bit of the byte.[37] */ + + // Pre-processing: padding with zeros + // append "0" bit until message length in bit ≡ 448 (mod 512) + // append length mod (2 pow 64) to message + + U32 new_len; + for (new_len = initial_len * 8 + 1; new_len % 512 != 448; new_len++) + ; + new_len /= 8; + + msg = CAlloc(new_len + 64, erythros_mem_task); // also appends "0" bits + // (we alloc also 64 extra bytes...) + MemCpy(msg, initial_msg, initial_len); + msg[initial_len] = 128; // write the "1" bit + + U32 bits_len = 8 * initial_len; // note, we append the len + MemCpy(msg + new_len, &bits_len, 4); // in bits at the end of the buffer + + // Process the message in successive 512-bit chunks: + // for each 512-bit chunk of message: + U32 offset; + for (offset = 0; offset < new_len; offset += (512 / 8)) { + + // break chunk into sixteen 32-bit words w[j], 0 ≤ j ≤ 15 + U32* w = (msg + offset)(U32*); + + // Initialize hash value for this chunk: + U32 a = md5_h0; + U32 b = md5_h1; + U32 c = md5_h2; + U32 d = md5_h3; + + // Main loop: + U32 i; + for (i = 0; i < 64; i++) { + + U32 f, g; + + if (i < 16) { + f = (b & c) | ((~b) & d); + g = i; + } else if (i < 32) { + f = (d & b) | ((~d) & c); + g = (5 * i + 1) % 16; + } else if (i < 48) { + f = b ^ c ^ d; + g = (3 * i + 5) % 16; + } else { + f = c ^ (b | (~d)); + g = (7 * i) % 16; + } + + U32 temp = d; + d = c; + c = b; + // printf("rotateLeft(%x + %x + %x + %x, %d)\n", a, f, k[i], w[g], r[i]); + b = b + LEFTROTATE((a + f + md5_k[i] + w[g]), md5_r[i]); + a = temp; + } + + // Add this chunk's hash to result so far: + + md5_h0 += a; + md5_h1 += b; + md5_h2 += c; + md5_h3 += d; + } + + md5_h[0] = md5_h0; + md5_h[1] = md5_h1; + md5_h[2] = md5_h2; + md5_h[3] = md5_h3; + + // cleanup + Free(msg); +} + +U8* md5_string(U8* buf, I64 size) +{ + U32 md5_h[4]; + md5(buf, size, &md5_h[0]); + U8* str = CAlloc(33, erythros_mem_task); + StrPrint(str + StrLen(str), "%02x%02x%02x%02x", md5_h[0].u8[0], + md5_h[0].u8[1], md5_h[0].u8[2], md5_h[0].u8[3]); + StrPrint(str + StrLen(str), "%02x%02x%02x%02x", md5_h[1].u8[0], + md5_h[1].u8[1], md5_h[1].u8[2], md5_h[1].u8[3]); + StrPrint(str + StrLen(str), "%02x%02x%02x%02x", md5_h[2].u8[0], + md5_h[2].u8[1], md5_h[2].u8[2], md5_h[2].u8[3]); + StrPrint(str + StrLen(str), "%02x%02x%02x%02x", md5_h[3].u8[0], + md5_h[3].u8[1], md5_h[3].u8[2], md5_h[3].u8[3]); + return str; +} diff --git a/System/Api/NetInfo.HC b/System/Api/NetInfo.HC new file mode 100644 index 0000000..79f0384 --- /dev/null +++ b/System/Api/NetInfo.HC @@ -0,0 +1,32 @@ +#define NETINFO_REQUEST_PTR 0x300030 + +MemSet(NETINFO_REQUEST_PTR, NULL, sizeof(U64)); + +class NetInfoRequest { + U64 mac_address; + U64 ipv4_address; + U64 ipv4_netmask; + U64 ipv4_network; + U64 ipv4_gateway; + U64 dns_server_address; + U64 dns_server_port; + U64 rx_bytes; + U64 rx_frames; + U64 tx_bytes; + U64 tx_frames; + U64 pointer_to_u32; +}; + +NetInfoRequest* @net_info_request() +{ + U32 res = 0; + NetInfoRequest* req = CAlloc(sizeof(NetInfoRequest), Fs->code_heap); + req->pointer_to_u32 = &res; + U64* request_ptr = NETINFO_REQUEST_PTR; + while (*request_ptr) + Sleep(1); + LXchgU32(request_ptr, req); + while (!res) + Sleep(1); + return req; +} diff --git a/System/Api/Tcp.HC b/System/Api/Tcp.HC new file mode 100644 index 0000000..99f436c --- /dev/null +++ b/System/Api/Tcp.HC @@ -0,0 +1,210 @@ +#define TCP_SOCKET_REQUEST_PTR 0x300000 +#define TCP_BIND_REQUEST_PTR 0x300040 +#define TCP_ACCEPT_REQUEST_PTR 0x300050 + +MemSet(TCP_SOCKET_REQUEST_PTR, NULL, sizeof(U64)); + +// TcpSocket states + +#define TCP_SOCKET_STATE_IDLE 0 +#define TCP_SOCKET_STATE_ESTABLISHED 1 +#define TCP_SOCKET_STATE_CLOSED 2 +#define TCP_SOCKET_STATE_CONNECTING 4 + +class TcpSocket { + U64 remote_addr; + U64 remote_port; + U64 state; + U64 receive_buffer_ptr; // Pointer to receive buffer in physical memory + U64 receive_buffer_size; + U64 receive_buffer_filled; // Number of bytes Net has put into buffer + U64 receive_buffer_kick; // Net sets this to 1 when it has data available for + // us, we set back to 0 when ready to receive + U64 send_buffer_ptr; + U64 send_buffer_size; + U64 send_buffer_filled; + U64 send_buffer_kick; // We set this to 1 when we have data available to net, + // Net sets back to 0 when ready to receive + U0(*close) + (); + U64(*receive) + (U64 buf, U64 length); + U0(*send) + (U64 buf, U64 length); +}; + +class TcpBind { + U64 port; + U64 function; + U64 response_code; +}; + +U8 @tcp_close_wrapper_function[16] + = { 0x55, 0x48, 0x8B, 0xEC, 0x68, 0x78, + 0x56, 0x34, 0x12, 0xE8, 0x02, 0x6D, + 0x02, 0x00, 0x5D, 0xC3 }; + +U8 @tcp_receive_wrapper_function[32] = { + 0x55, 0x48, 0x8B, 0xEC, 0x56, 0x57, 0x48, 0x8B, 0x75, 0x18, 0x48, + 0x8B, 0x7D, 0x10, 0x56, 0x57, 0x68, 0x78, 0x56, 0x34, 0x12, 0xE8, + 0x5E, 0x62, 0x02, 0x00, 0x5F, 0x5E, 0x5D, 0xC2, 0x10, 0x00 +}; + +U8 @tcp_send_wrapper_function[32] = { + 0x55, 0x48, 0x8B, 0xEC, 0x56, 0x57, 0x48, 0x8B, 0x75, 0x18, 0x48, + 0x8B, 0x7D, 0x10, 0x56, 0x57, 0x68, 0x78, 0x56, 0x34, 0x12, 0xE8, + 0x5E, 0x62, 0x02, 0x00, 0x5F, 0x5E, 0x5D, 0xC2, 0x10, 0x00 +}; + +U0 @tcp_socket_send(TcpSocket* s, U64 buf, U64 length) +{ + while (s->send_buffer_kick) + Sleep(1); + U64 pos = 0; + U64 bytes_to_send = 0; + while (pos < length) { + if ((length - pos) > s->send_buffer_size) + bytes_to_send = s->send_buffer_size; + else + bytes_to_send = length - pos; + MemCpy(s->send_buffer_ptr, buf + pos, bytes_to_send); + s->send_buffer_filled = bytes_to_send; + s->send_buffer_kick = 1; + pos += bytes_to_send; + while (s->send_buffer_kick) + Sleep(1); + } +} + +U64 @tcp_socket_receive(TcpSocket* s, U64 buf, U64 size) +{ + s->receive_buffer_size = size; + s->receive_buffer_kick = 0; + while (!s->receive_buffer_kick) { + if (s->state == TCP_SOCKET_STATE_CLOSED) + return NULL; + Sleep(1); + } + U64 bytes_received = s->receive_buffer_filled; + if (bytes_received > 0) { + MemCpy(buf, s->receive_buffer_ptr, bytes_received); + } + return bytes_received; +} + +U0 @tcp_wait_for_connection_established(TcpSocket* s) +{ + while (s->state != TCP_SOCKET_STATE_ESTABLISHED) + Sleep(1); +} + +U0 @tcp_socket_close(TcpSocket* s) +{ + if (s->close) + Free(s->close); + if (s->receive) + Free(s->receive); + if (s->send) + Free(s->send); + s->state = TCP_SOCKET_STATE_CLOSED; +} + +TcpSocket* @tcp_socket_create(U8* host, U64 port) +{ + U64 addr = @dns_query(host); + TcpSocket* s = CAlloc(sizeof(TcpSocket), erythros_mem_task->code_heap); + s->remote_addr = addr; + s->remote_port = port; + + U64 a; + + s->close = MAlloc(16, erythros_mem_task->code_heap); + MemCpy(s->close, @tcp_close_wrapper_function, 16); + a = s->close; + a += 0x05; + MemSetU32(a, s, 1); + a = s->close; + a += 0x09; + @patch_call_rel32(a, &@tcp_socket_close); + + s->receive = MAlloc(25, erythros_mem_task->code_heap); + MemCpy(s->receive, @tcp_receive_wrapper_function, 32); + a = s->receive; + a += 0x11; + MemSetU32(a, s, 1); + a = s->receive; + a += 0x15; + @patch_call_rel32(a, &@tcp_socket_receive); + + s->send = MAlloc(32, erythros_mem_task->code_heap); + MemCpy(s->send, @tcp_send_wrapper_function, 32); + a = s->send; + a += 0x11; + MemSetU32(a, s, 1); + a = s->send; + a += 0x15; + @patch_call_rel32(a, &@tcp_socket_send); + + U64* request_ptr = TCP_SOCKET_REQUEST_PTR; + while (*request_ptr) + Sleep(1); + LXchgU32(request_ptr, s); + return s; +} + +U64 @tcp_socket_bind(U64 port, U64 function) +{ + if (!port || !function) + return NULL; + + TcpBind* b = CAlloc(sizeof(TcpBind), erythros_mem_task->code_heap); + b->port = port; + b->function = function; // U0 my_spawn_wrapper_function(TcpSocket* s) + + U64* request_ptr = TCP_BIND_REQUEST_PTR; + while (*request_ptr) + Sleep(1); + LXchgU32(request_ptr, b); + while (*request_ptr) + Sleep(1); + U64 res = b->response_code; + Free(b); + return res; +} + +TcpSocket* @tcp_socket_accept(TcpSocket* s) +{ + if (!s || !s->remote_addr || !s->remote_port) + return NULL; + + U64 a; + + s->close = MAlloc(16, erythros_mem_task->code_heap); + MemCpy(s->close, @tcp_close_wrapper_function, 16); + a = s->close; + a += 0x05; + MemSetU32(a, s, 1); + a = s->close; + a += 0x09; + @patch_call_rel32(a, &@tcp_socket_close); + + s->receive = MAlloc(25, erythros_mem_task->code_heap); + MemCpy(s->receive, @tcp_receive_wrapper_function, 32); + a = s->receive; + a += 0x11; + MemSetU32(a, s, 1); + a = s->receive; + a += 0x15; + @patch_call_rel32(a, &@tcp_socket_receive); + + s->send = MAlloc(32, erythros_mem_task->code_heap); + MemCpy(s->send, @tcp_send_wrapper_function, 32); + a = s->send; + a += 0x11; + MemSetU32(a, s, 1); + a = s->send; + a += 0x15; + @patch_call_rel32(a, &@tcp_socket_send); + + return s; +} diff --git a/System/Api/Tls.HC b/System/Api/Tls.HC new file mode 100644 index 0000000..0d51f21 --- /dev/null +++ b/System/Api/Tls.HC @@ -0,0 +1,99 @@ +#define TLS_CONNECT_TASK_STACK_SIZE 524288 +#define TLS_CLIENT_MESSAGE_BUFFER_SIZE 0xFFFF + +class TlsSocket : TcpSocket { + U64 ctx; + U8 client_message[TLS_CLIENT_MESSAGE_BUFFER_SIZE]; +}; + +U0 @tls_send_pending(TlsSocket* s) +{ + U32 out_buffer_len = 0; + U8* out_buffer = @tls_get_write_buffer(s->ctx, &out_buffer_len); + if (out_buffer && out_buffer_len) { + @tcp_socket_send(s, out_buffer, out_buffer_len); + @tls_buffer_clear(s->ctx); + } +} + +U0 @tls_socket_send(TlsSocket* s, U64 buf, U64 size) +{ + @tls_write(s->ctx, buf, size); + @tls_send_pending(s); +} + +U64 @tls_socket_receive(TlsSocket* s, U8* buf, I64 size) +{ + I64 len = @tcp_socket_receive(s, s->client_message, TLS_CLIENT_MESSAGE_BUFFER_SIZE); + if (len) { + @tls_consume_stream(s->ctx, s->client_message, len, NULL); + @tls_send_pending(s); + } + return @tls_read(s->ctx, buf, size); +} + +U0 @tls12_connect(TlsSocket* s) +{ + I64 len; + @tls_client_connect(s->ctx); + @tls_send_pending(s); + while (!@tls_established(s->ctx)) { + len = @tcp_socket_receive(s, &s->client_message, TLS_CLIENT_MESSAGE_BUFFER_SIZE); + if (len) { + @tls_consume_stream(s->ctx, &s->client_message, len, NULL); + @tls_send_pending(s); + } + Sleep(1); + } +} + +TlsSocket* @tls_socket_create(U8* server_name, U64 port = 443) +{ + U64 addr = @dns_query(server_name); + TlsSocket* s = CAlloc(sizeof(TlsSocket), erythros_mem_task->code_heap); + s->remote_addr = addr; + s->remote_port = port; + + U64 a; + + s->close = MAlloc(16, erythros_mem_task->code_heap); + MemCpy(s->close, @tcp_close_wrapper_function, 16); + a = s->close; + a += 0x05; + MemSetU32(a, s, 1); + a = s->close; + a += 0x09; + @patch_call_rel32(a, &@tcp_socket_close); + + s->receive = MAlloc(25, erythros_mem_task->code_heap); + MemCpy(s->receive, @tcp_receive_wrapper_function, 32); + a = s->receive; + a += 0x11; + MemSetU32(a, s, 1); + a = s->receive; + a += 0x15; + @patch_call_rel32(a, &@tls_socket_receive); + + s->send = MAlloc(32, erythros_mem_task->code_heap); + MemCpy(s->send, @tcp_send_wrapper_function, 32); + a = s->send; + a += 0x11; + MemSetU32(a, s, 1); + a = s->send; + a += 0x15; + @patch_call_rel32(a, &@tls_socket_send); + + U64* request_ptr = TCP_SOCKET_REQUEST_PTR; + while (*request_ptr) + Sleep(1); + LXchgU32(request_ptr, s); + + while (s->state != TCP_SOCKET_STATE_ESTABLISHED) + Sleep(1); + + s->ctx = @tls_create_context(0, TLS_V12); + @tls_sni_set(s->ctx, StrNew(server_name, erythros_mem_task->code_heap)); + Spawn(&@tls12_connect, s, , , , TLS_CONNECT_TASK_STACK_SIZE); + + return s; +} diff --git a/System/Config/Net.json b/System/Config/Net.json new file mode 100644 index 0000000..9217064 --- /dev/null +++ b/System/Config/Net.json @@ -0,0 +1,10 @@ +{ + "tcpip.ipv4_address": "10.20.0.10", + "tcpip.ipv4_netmask": "255.255.255.0", + "tcpip.ipv4_network": "10.20.0.0", + "tcpip.ipv4_gateway": "10.20.0.254", + "tcpip.ipv4_dns_server_address": "8.8.8.8", + "tcpip.ipv4_dns_server_port": "53", + "tcpip.mss_size": "1360", + "eof": "eof" +} \ No newline at end of file diff --git a/System/Core/Compositor.HC b/System/Core/Compositor.HC new file mode 100644 index 0000000..1a3da84 --- /dev/null +++ b/System/Core/Compositor.HC @@ -0,0 +1,1112 @@ +#define CPZ_MSG_WIN_CREATE 0x1001 +#define CPZ_MSG_WIN_DESTROY 0x1002 +#define CPZ_MSG_WIN_REPAINT 0x1003 +#define CPZ_MSG_WIN_MOVE_TO 0x1004 +#define CPZ_MSG_WIN_RESIZE_TO 0x1005 +#define CPZ_MSG_WIN_MOUSE_AT 0x1006 +#define CPZ_MSG_WIN_MOUSE_WHEEL 0x1007 +#define CPZ_MSG_WIN_LEFT_BTN_UP 0x1008 +#define CPZ_MSG_WIN_LEFT_BTN_DOWN 0x1009 +#define CPZ_MSG_WIN_RIGHT_BTN_UP 0x100A +#define CPZ_MSG_WIN_RIGHT_BTN_DOWN 0x100B +#define CPZ_MSG_WIN_KEY_PRESS 0x100C +#define CPZ_MSG_WIN_WIDGET_DESTROY 0x100D +#define CPZ_MSG_WIN_SET_Z_INDEX 0x100E +#define CPZ_MSG_SET_WALLPAPER 0x100F + +#define CPZ_WALLPAPER_CENTERED 0x0 +#define CPZ_WALLPAPER_AUTORESIZE 0x1 +#define CPZ_WALLPAPER_REPEAT 0x2 + +class @compositor_menubar +{ + Window* win; + CTask* task; + TextInputWidget* title; +}; + +class @compositor_windows_list +{ + @compositor_windows_list* prev; + @compositor_windows_list* next; + Window* window; +}; + +class @compositor +{ + I64 next_id; + I64 max_z_index; + Window* active_win; + Context2D* blend_ctx; + Context2D* ctx; + Context2D* pointer; + @compositor_menubar menubar; + @compositor_windows_list* windows; + @compositor_windows_list* global_input_event_listeners; + Bounds2D bounds; + @mouse mouse; + @session session; + @theme theme; + Bool in_drag; + Bool in_resize; + CTask* task; + U0(*Init) + (); + Window* (*CreateWindow)(I64 x, I64 y, I64 width, I64 height, + I64 flags = WIN_FLAGS_DEFAULT, U8* title = NULL, + Context2D* icon = NULL); + U0 (*DestroyWindow)(Window* win); + Window (*GetWindowByTitle)(U8* title); + Window (*GetWindowByZIndex)(I64 index); + U0 (*HideWindow)(Window* win); + U0 (*ShowWindow)(Window* win); + U0 (*RegisterForGlobalInputEvents)(Window* win); + U0 (*UnregisterForGlobalInputEvents)(Window* win); + U0 (*SetWallpaper)(Context2D* ctx, U32 mode = CPZ_WALLPAPER_AUTORESIZE, U32 background = Color(0, 0, 0)); + U0(*Task) + (); +}; + +@compositor Compositor; + +U0 @compositor_add_window_to_list(Window* win) +{ + @compositor_windows_list* win_list = Compositor.windows; + @compositor_windows_list* win_next = CAlloc(sizeof(@compositor_windows_list)); + while (win_list->next) { + win_list = win_list->next; + } + win_next->window = win; + win_next->prev = win_list; + win_list->next = win_next; +} + +U0 @compositor_add_global_input_event_listener_to_list(Window* win) +{ + @compositor_windows_list* win_list = Compositor.global_input_event_listeners; + @compositor_windows_list* win_next = CAlloc(sizeof(@compositor_windows_list)); + while (win_list->next) { + win_list = win_list->next; + } + win_next->window = win; + win_next->prev = win_list; + win_list->next = win_next; +} + +U0 @compositor_remove_global_input_event_listener_from_list(Window* win) +{ + @compositor_windows_list* win_list = Compositor.global_input_event_listeners; + @compositor_windows_list* win_list_prev = NULL; + @compositor_windows_list* win_list_next = NULL; + + while (win_list) { + if (win_list->window == win) { + win_list_prev = win_list->prev; + win_list_next = win_list->next; + win_list_prev->next = win_list_next; + win_list_next->prev = win_list_prev; + Free(win_list); + return; + } + win_list = win_list->next; + } +} + +U0 @compositor_refresh(Window* win) +{ + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + msg->client = win->client; + msg->type = CPZ_MSG_WIN_REPAINT; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); +} + +U0 @compositor_set_z_index(Window* win, I64 index) +{ + if (!win) + return; + I64 i = 0; + @compositor_windows_list* win_index = NULL; + @compositor_windows_list* win_list = Compositor.windows->next; + @compositor_windows_list* prev = NULL; + @compositor_windows_list* next = NULL; + while (win_list) { + if (win_list->window == win) { + win_index = win_list; + if (win_index->prev) + prev = win_index->prev; + if (win_index->next) + next = win_index->next; + if (prev) + prev->next = next; + if (next) + next->prev = prev; + break; + } + win_list = win_list->next; + } + + win_list = Compositor.windows->next; + prev = NULL; + + while (win_list) { + if (i == index) { + prev = win_list->prev; + if (prev) + prev->next = win_index; + win_index->prev = prev; + win_index->next = win_list; + win_list->prev = win_index; + break; + } + i++; + win_list = win_list->next; + } +} + +U0 @compositor_set_z_index_send_msg(Window* win, I64 index) +{ + if (!win) + return; + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + msg->client = win->client; + msg->type = CPZ_MSG_WIN_SET_Z_INDEX; + msg->payload = win; + msg->i64 = index; + Ipc.MsgSend(Compositor.task, msg); +} + +U0 @compositor_set_wallpaper(Context2D* ctx, U32 mode, U32 background) +{ // Sets the wallpaper, doesn't Free the Context, you need to Free it after. + if (!ctx) { + return; + } + Window* win = Compositor.GetWindowByTitle("Wallpaper"); + Context2D* tmp = ctx; + + if (!win) { // Please, don't call this if the system isn't initialized + System.Log(Fs, "Trying to set the Wallpaper, but the System isn't initialized completely!"); + return; + } + + if (Display.Width() != ctx->width || Display.Height() != ctx->height) { + if (Display.Width() > ctx->width && Display.Height() > ctx->height) { + switch (mode) { + case CPZ_WALLPAPER_CENTERED: + tmp = NewContext2D(Display.Width(), Display.Height()); + Fill2D(tmp, background); + CopyRect2D(tmp, + (Display.Width() / 2) - (ctx->width / 2), + (Display.Height() / 2) - (ctx->height / 2), + ctx); + break; + case CPZ_WALLPAPER_REPEAT: + tmp = NewContext2D(Display.Width(), Display.Height()); + I64 x = 0; + I64 y = 0; + while (TRUE) { + CopyRect2D(tmp, x, y, ctx); + x += ctx->width; + if (x >= Display.Width()) { + x = 0; + y += ctx->height; + if (y >= Display.Height()) + break; + } + } + break; + case CPZ_WALLPAPER_AUTORESIZE: + default: + tmp = Scale2D(tmp, Display.Width() / ToF64(ctx->width), Display.Height() / ToF64(ctx->height)); + break; + } + } else { + tmp = Scale2D(tmp, Display.Width() / ToF64(ctx->width), Display.Height() / ToF64(ctx->height)); + } + } + + MemCpyU32(win->backing_store->fb, tmp->fb, + Display.Width() * Display.Height()); + if (tmp != ctx) // We don't want to delete the context so it can be reused later (If necessary) + DelContext2D(tmp); +} + +U0 @compositor_set_wallpaper_send_msg(Context2D* ctx, U32 mode = CPZ_WALLPAPER_AUTORESIZE, U32 background = Color(0, 0, 0)) +{ + if (!ctx) + return; + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + msg->client = Fs; + msg->type = CPZ_MSG_SET_WALLPAPER; + msg->payload = ctx; + msg->i64.u32[0] = mode; + msg->i64.u32[1] = background; + Ipc.MsgSend(Compositor.task, msg); +} + +U0 @compositor_ipc_queue_process() +{ + Window* win; + IpcMessage* msg; + @compositor_windows_list* win_list = Compositor.windows->next; + @compositor_windows_list* prev; + @compositor_windows_list* next; + msg = Ipc.MsgRecv(); + if (msg) { + switch (msg->type) { + case CPZ_MSG_WIN_CREATE: + win = msg->payload; + win->client = msg->client; + win->render_ctx = NewContext2D(Display.Width(), Display.Height()); + win->backing_store = NewContext2D(Display.Width(), Display.Height()); + win->backing_store->width = win->width; + win->backing_store->height = win->height; + @compositor_add_window_to_list(win); + System.Log(Fs, "Received message ← CreateWindow (%dx%d at %d, %d)", + win->width, win->height, win->x, win->y); + Free(msg); + msg = CAlloc(sizeof(IpcMessage)); + msg->client = win->client; + msg->type = CPZ_MSG_WIN_REPAINT; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + break; + case CPZ_MSG_WIN_DESTROY: + win = msg->payload; + while (win_list) { + if (win_list->window == win) { + prev = win_list->prev; + next = win_list->next; + prev->next = next; + next->prev = prev; + if (win_list->window->backing_store) + DelContext2D(win_list->window->backing_store); + if (win_list->window->render_ctx) + DelContext2D(win_list->window->render_ctx); + // FIXME: free Widgets + Free(win_list->window); + Free(win_list); + Compositor.active_win = NULL; + System.Log(Fs, "Received message ← DestroyWindow 0x%08x", win); + break; + } + win_list = win_list->next; + } + break; + case CPZ_MSG_WIN_SET_Z_INDEX: + @compositor_set_z_index(msg->payload, msg->i64); + System.Log(Fs, "Received message ← SetZIndex (%08X, %d)", msg->payload, + msg->i64); + Free(msg); + break; + case CPZ_MSG_SET_WALLPAPER: + Context2D* ctx = msg->payload; + U32 mode = msg->i64.u32[0]; + U32 background = msg->i64.u32[1]; + System.Log(Fs, "Received message ← SetWallpaper (%08X, Mode: %d, Background: %d)", ctx, + mode, background); + @compositor_set_wallpaper(ctx, mode, background); + Free(msg); + break; + default: + Free(msg); + break; + } + } +} + +Window* @compositor_get_window_by_title(U8* title) +{ + @compositor_windows_list* win_list = Compositor.windows->next; + while (win_list) { + if (win_list->window) + if (!StrCmp(&win_list->window->title, title)) + return win_list->window; + win_list = win_list->next; + } + return NULL; +} + +Window* @compositor_get_window_by_z_index(I64 index) +{ + @compositor_windows_list* win_list = Compositor.windows->next; + I64 i = 0; + while (win_list) { + if (i == index) + return win_list->window; + i++; + win_list = win_list->next; + } + return NULL; +} + +U0 @compositor_handle_active_window_flags() +{ + if (!Compositor.active_win) + return; + if (Compositor.active_win->flags & WIN_FLAGS_MINIMIZED) { + Compositor.active_win = Compositor.GetWindowByTitle("Wallpaper"); + if (Compositor.menubar.win) { + Gui.Widget.SetText(Compositor.menubar.title, + &Compositor.active_win->title); + Gui.Window.Refresh(Compositor.menubar.win); + } + } +} + +Context2D* @compositor_set_theme_pointer(U8* path, U8* pointer) +{ + U8 file_path[512]; + StrPrint(&file_path, "%sPointer/%s.png", path, pointer); + Context2D* ctx = Image.FileToContext2D(&file_path); + return ctx; +} + +U0 @compositor_set_theme_wallpaper(U8* path) +{ + U8 file_path[512]; + StrPrint(&file_path, "%s%s", path, "wallpaper.jpg"); + Compositor.theme.wallpaper = Image.FileToContext2D(file_path); + if (!Compositor.theme.wallpaper) { + Compositor.theme.wallpaper = NewContext2D(Display.Width(), Display.Height()); + } +} + +U0 @compositor_set_theme(U8* theme) +{ + U8 theme_path[512]; + U8 exec_path[512]; + StrPrint(&theme_path, "M:/Media/Themes/%s/", theme); + if (!IsDir(&theme_path)) { + System.Log(Fs, "SetTheme failed: Theme does not exist: '%s'", theme); + return; + } + // FIXME: This is disgusting + Compositor.theme.pointer.pointer = + @compositor_set_theme_pointer(&theme_path, "pointer"); + Compositor.theme.pointer.pen = + @compositor_set_theme_pointer(&theme_path, "pen"); + Compositor.theme.pointer.move = + @compositor_set_theme_pointer(&theme_path, "move"); + Compositor.theme.pointer.link = + @compositor_set_theme_pointer(&theme_path, "link"); + Compositor.theme.pointer.horz = + @compositor_set_theme_pointer(&theme_path, "horz"); + Compositor.theme.pointer.vert = + @compositor_set_theme_pointer(&theme_path, "vert"); + Compositor.theme.pointer.text = + @compositor_set_theme_pointer(&theme_path, "text"); + Compositor.theme.pointer.cross = + @compositor_set_theme_pointer(&theme_path, "cross"); + Compositor.theme.pointer.dgn1 = + @compositor_set_theme_pointer(&theme_path, "dgn1"); + Compositor.theme.pointer.dgn2 = + @compositor_set_theme_pointer(&theme_path, "dgn2"); + Compositor.theme.pointer.help = + @compositor_set_theme_pointer(&theme_path, "help"); + Compositor.theme.pointer.alternate = + @compositor_set_theme_pointer(&theme_path, "alternate"); + Compositor.theme.pointer.unavailable = + @compositor_set_theme_pointer(&theme_path, "unavailable"); + + @compositor_set_theme_wallpaper(&theme_path); + + StrPrint(&exec_path, "%sTheme.HC", &theme_path); + ExeDoc(DocRead(exec_path)); +} + +Bool @compositor_active_win_flag_is_set(U64 flag) +{ + if (@gui_window_flag_is_set(Compositor.active_win, flag)) + return TRUE; + return FALSE; +} + +U0 @compositor_set_pointer(Context2D* pointer = NULL) +{ + if (!pointer) + pointer = Compositor.theme.pointer.pointer; + if (Animation2D.IsAnimation(pointer)) + Compositor.pointer = Animation2D.Frame(pointer); + else + Compositor.pointer = pointer; + Mouse.PointerSet(Compositor.pointer->fb, Compositor.pointer->width, + Compositor.pointer->height); +} + +U0 @compositor_set_active_window(Window* win) +{ + if (!win) + return; + if (Compositor.active_win == win) + return; + IpcMessage* msg; + @compositor_windows_list* win_list = Compositor.windows->next; + @compositor_windows_list* prev; + @compositor_windows_list* next; + while (win_list) { + if (!(win->flags & WIN_FLAGS_NO_REINDEX)) { + if (win_list->window == win && win_list->next) { + prev = win_list->prev; + next = win_list->next; + prev->next = next; + next->prev = prev; + while (next->next) { + next = next->next; + } + win_list->prev = next; + win_list->next = NULL; + next->next = win_list; + break; + } + } + win_list = win_list->next; + } + if (Compositor.active_win && Compositor.active_win != win) { + msg = CAlloc(sizeof(IpcMessage)); + msg->client = Compositor.active_win->client; + msg->type = CPZ_MSG_WIN_REPAINT; + msg->payload = Compositor.active_win; + Ipc.MsgSend(msg->client, msg); + } + Compositor.active_win = win; + if (Compositor.menubar.win && Compositor.menubar.title) { + Gui.Widget.SetText(Compositor.menubar.title, win->title); + Gui.Window.Refresh(Compositor.menubar.win); + } + System.Log(Fs, "SetActiveWindow (%dx%d at %d, %d)", win->width, win->height, + win->x, win->y); + msg = CAlloc(sizeof(IpcMessage)); + msg->client = win->client; + msg->type = CPZ_MSG_WIN_REPAINT; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + @compositor_set_pointer(); +} + +U0 @compositor_handle_window_resize() +{ + if (!Compositor.active_win) + return; + if (Compositor.active_win->signature != WIN_SIGNATURE) + return; + if (!(@compositor_active_win_flag_is_set(WIN_FLAGS_RESIZABLE))) + goto resize_set_pointer; + + Bool set_pointer_to_resize = FALSE; + I64 new_width; + I64 new_height; + + // Bottom right + if (Mouse.x > Compositor.active_win->x + Compositor.active_win->width - 16 && Mouse.x < Compositor.active_win->x + Compositor.active_win->width && Mouse.y > Compositor.active_win->y + Compositor.active_win->height - 16 && Mouse.y < Compositor.active_win->y + Compositor.active_win->height) { + @compositor_set_pointer(Compositor.theme.pointer.dgn1); + set_pointer_to_resize = TRUE; + } + + if (Mouse.left && !Compositor.mouse.left && Compositor.active_win) { + if (Mouse.x > Compositor.active_win->x + Compositor.active_win->width - 16 && Mouse.x < Compositor.active_win->x + Compositor.active_win->width && Mouse.y > Compositor.active_win->y + Compositor.active_win->height - 16 && Mouse.y < Compositor.active_win->y + Compositor.active_win->height) { + Compositor.active_win->origin.x = Compositor.active_win->x; + Compositor.active_win->origin.y = Compositor.active_win->y; + Compositor.active_win->origin.width = Compositor.active_win->width; + Compositor.active_win->origin.height = Compositor.active_win->height; + Compositor.active_win->origin.mouse_x = Mouse.x; + Compositor.active_win->origin.mouse_y = Mouse.y; + Compositor.in_resize = TRUE; + } + } + + if (!Mouse.left) { + Compositor.in_drag = FALSE; + Compositor.in_resize = FALSE; + } + + if (Compositor.in_resize) { + // FIXME: Set minimum width and height in Compositor.theme or Window. + new_width = Max(16, Compositor.active_win->origin.width + (Mouse.x - Compositor.active_win->origin.mouse_x)); + new_height = Max(16, Compositor.active_win->origin.height + (Mouse.y - Compositor.active_win->origin.mouse_y)); + new_width = Max(Compositor.theme.window.min_width, new_width); + new_height = Max(Compositor.theme.window.min_height, new_height); + if (Compositor.active_win->width != new_width || Compositor.active_win->height != new_height && !Compositor.active_win->repainting) { + Window* win = Compositor.active_win; + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + System.Log(Fs, + "Sent message → WindowResizeTo (%dx%d at %d, %d) to (%dx%d)", + Compositor.active_win->width, Compositor.active_win->height, + Compositor.active_win->x, Compositor.active_win->y, new_width, + new_height); + + if (win->signature != WIN_SIGNATURE) + return; + + win->width = new_width; + win->height = new_height; + msg->client = Compositor.active_win->client; + msg->type = CPZ_MSG_WIN_RESIZE_TO; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + } + } + +resize_set_pointer: + // FIXME: Move this to @compositor_set_pointer_for_window() or somewhere more + // appropriate? + if (!set_pointer_to_resize) { + if (Compositor.active_win) + if (@gui_window_is_hovered(Compositor.active_win)) + if (Compositor.active_win->hovered_widget) + if (Compositor.active_win->hovered_widget->pointer) + @compositor_set_pointer( + Compositor.active_win->hovered_widget->pointer); + else { + if (Compositor.active_win->pointer) + @compositor_set_pointer(Compositor.active_win->pointer); + else + @compositor_set_pointer(); + } + else { + if (Compositor.active_win->pointer) + @compositor_set_pointer(Compositor.active_win->pointer); + else + @compositor_set_pointer(); + } + else + @compositor_set_pointer(); + else + @compositor_set_pointer(); + } +} + +U0 @compositor_handle_window_drag() +{ + if (!(@compositor_active_win_flag_is_set(WIN_FLAGS_MOVABLE))) + return; + + I64 title_length; + I64 title_offset; + I64 new_x; + I64 new_y; + + // FIXME: Get title_length and title_offset values from theme and Window + // flags. + title_offset = 0; + title_length = Compositor.active_win->width; + + if (@compositor_active_win_flag_is_set(WIN_FLAGS_TITLE_BAR)) { + title_offset = Compositor.active_win->title_bar_x; + title_length = Compositor.active_win->title_bar_width; + } + + if (Mouse.left && !Compositor.mouse.left && Compositor.active_win) { + if (Mouse.x > Compositor.active_win->x + title_offset && Mouse.x < Compositor.active_win->x + title_offset + title_length && Mouse.y > Compositor.active_win->y && Mouse.y < Compositor.active_win->y + 18) { + Compositor.active_win->origin.x = Compositor.active_win->x; + Compositor.active_win->origin.y = Compositor.active_win->y; + Compositor.active_win->origin.width = Compositor.active_win->width; + Compositor.active_win->origin.height = Compositor.active_win->height; + Compositor.active_win->origin.mouse_x = Mouse.x; + Compositor.active_win->origin.mouse_y = Mouse.y; + Compositor.in_drag = TRUE; + } + } + + if (!Mouse.left) { + Compositor.in_drag = FALSE; + Compositor.in_resize = FALSE; + } + + if (Compositor.in_drag) { + new_x = Compositor.active_win->origin.x + (Mouse.x - Compositor.active_win->origin.mouse_x); + new_y = Compositor.active_win->origin.y + (Mouse.y - Compositor.active_win->origin.mouse_y); + if (Compositor.active_win->x != new_x || Compositor.active_win->y != new_y) { + Window* win = Compositor.active_win; + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + System.Log( + Fs, "Sent message → WindowMoveTo (%dx%d at %d, %d) to (%d, %d)", + Compositor.active_win->width, Compositor.active_win->height, + Compositor.active_win->x, Compositor.active_win->y, new_x, new_y); + win->x = Min(Compositor.bounds.x2, Max(Compositor.bounds.x1, new_x)); + win->y = Min(Compositor.bounds.y2, Max(Compositor.bounds.y1, new_y)); + msg->client = Compositor.active_win->client; + msg->type = CPZ_MSG_WIN_MOVE_TO; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + } + } +} + +U0 @compositor_handle_window_select() +{ + Window* win; + @compositor_windows_list* win_list; + if (Mouse.left && !Compositor.mouse.left) { + if (((Mouse.x < Compositor.active_win->x || Mouse.x > Compositor.active_win->x + Compositor.active_win->width) || (Mouse.y < Compositor.active_win->y || Mouse.y > Compositor.active_win->y + Compositor.active_win->height)) || + @compositor_active_win_flag_is_set(WIN_FLAGS_NO_REINDEX)) { + win_list = Compositor.windows->next; + while (win_list->next) { + win_list = win_list->next; + } + while (win_list) { + if (Mouse.x > win_list->window->x && Mouse.x < win_list->window->x + win_list->window->width && Mouse.y > win_list->window->y && Mouse.y < win_list->window->y + win_list->window->height) { + if (Compositor.active_win != win_list->window && !(@gui_window_flag_is_set(win_list->window, WIN_FLAGS_MINIMIZED)) && !(@gui_window_flag_is_set(win_list->window, WIN_FLAGS_HIDDEN))) { + @compositor_set_active_window(win_list->window); + return; + } + } + win_list = win_list->prev; + } + } + } +} + +U0 @compositor_handle_global_input_events() +{ + // FIXME: Handle registered global input events + IpcMessage* msg; + I64 mouse_x; + I64 mouse_y; + I64 type = NULL; + Bool mouse_left = Mouse.left; + Bool mouse_right = Mouse.right; + + I64 key = Keyboard.active_key; + I64 tS = Keyboard.active_key_tS; + + if (key && tS != Keyboard.last_key_tS) { + + @compositor_windows_list* win_list = Compositor.global_input_event_listeners->next; + while (win_list) { + msg = CAlloc(sizeof(IpcMessage)); + System.Log(Fs, "Sent message → WinKeyPress [%08x] to window 0x%08x", key, + win_list->window); + msg->client = win_list->window->client; + msg->type = CPZ_MSG_WIN_KEY_PRESS; + msg->payload = win_list->window; + msg->i64 = key; + Ipc.MsgSend(msg->client, msg); + win_list = win_list->next; + } + } +} + +U0 @compositor_register_global_input_event_listener(Window* win) +{ + @compositor_add_global_input_event_listener_to_list(win); +} + +U0 @compositor_unregister_global_input_event_listener(Window* win) +{ + @compositor_remove_global_input_event_listener_from_list(win); +} + +U0 @compositor_handle_window_input_events(Window* win) +{ + if (!win) + return; + + IpcMessage* msg; + I64 mouse_x; + I64 mouse_y; + I64 type = NULL; + Bool mouse_left = Mouse.left; + Bool mouse_right = Mouse.right; + + if (win->focused_widget) { + if (win->focused_widget->type == WIDGET_TYPE_INPUT) { + if (@widget_input_handle_key(win->focused_widget)) { + msg = CAlloc(sizeof(IpcMessage)); + System.Log(Fs, "Sent message → WinKeyPress"); + msg->client = win->client; + msg->type = CPZ_MSG_WIN_KEY_PRESS; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + } + if (win->focused_widget(BitmapFontTextInputWidget*) + ->selected_region_start + == -1 + || win->focused_widget(BitmapFontTextInputWidget*) + ->selected_region_end + == -1) { + if (win->focused_widget(BitmapFontTextInputWidget*)->blink != Blink) { + win->focused_widget(BitmapFontTextInputWidget*)->blink = !win->focused_widget(BitmapFontTextInputWidget*)->blink; + @compositor_refresh(win); + } + } + } + } + + if (Mouse.z != Compositor.mouse.delta_z) { + msg = CAlloc(sizeof(IpcMessage)); + System.Log(Fs, "Sent message → WindowMouseWheel"); + msg->client = win->client; + msg->type = CPZ_MSG_WIN_MOUSE_WHEEL; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + } + + if (Mouse.x != Compositor.mouse.x || Mouse.y != Compositor.mouse.y || Mouse.left != Compositor.mouse.left || Mouse.right != Compositor.mouse.right) { + if (Mouse.x != Compositor.mouse.x || Mouse.y != Compositor.mouse.y) { + mouse_x = Mouse.x - win->x; + mouse_y = Mouse.y - win->y; + if (Mouse.x >= win->x && Mouse.x <= win->x + win->width && Mouse.y >= win->y && Mouse.y <= win->y + win->height && (mouse_x != win->mouse.x || mouse_y != win->mouse.y)) { + win->mouse.x = mouse_x; + win->mouse.y = mouse_y; + msg = CAlloc(sizeof(IpcMessage)); + System.Log(Fs, "Sent message → WindowMouseAt (%dx%d)", mouse_x, + mouse_y); + msg->client = win->client; + msg->type = CPZ_MSG_WIN_MOUSE_AT; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + } + } + if (mouse_left != win->mouse.left) { + switch (mouse_left) { + case MS_UP: + System.Log(Fs, "Sent message → WindowMouseLeftBtnUp"); + type = CPZ_MSG_WIN_LEFT_BTN_UP; + break; + case MS_DOWN: + System.Log(Fs, "Sent message → WindowMouseLeftBtnDown"); + type = CPZ_MSG_WIN_LEFT_BTN_DOWN; + break; + default: + break; + } + win->mouse.left = mouse_left; + win->left_btn_down.x = Mouse.x - win->x; + win->left_btn_down.y = Mouse.y - win->y; + msg = CAlloc(sizeof(IpcMessage)); + msg->client = win->client; + msg->type = type; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + } + if (mouse_right != win->mouse.right) { + switch (mouse_right) { + case MS_UP: + System.Log(Fs, "Sent message → WindowMouseRightBtnUp"); + type = CPZ_MSG_WIN_RIGHT_BTN_UP; + break; + case MS_DOWN: + System.Log(Fs, "Sent message → WindowMouseRightBtnDown"); + type = CPZ_MSG_WIN_RIGHT_BTN_DOWN; + break; + default: + break; + } + win->mouse.right = mouse_right; + win->right_btn_down.x = Mouse.x - win->x; + win->right_btn_down.y = Mouse.y - win->y; + msg = CAlloc(sizeof(IpcMessage)); + msg->client = win->client; + msg->type = type; + msg->payload = win; + Ipc.MsgSend(msg->client, msg); + } + } +} + +U0 @compositor_handle_windows_input_events() +{ + if (Compositor.active_win) + @compositor_handle_window_input_events(Compositor.active_win); + @compositor_windows_list* win_list = Compositor.windows; + Compositor.max_z_index = 0; + while (win_list) { + if (win_list->window) { + if (win_list->window->flags & WIN_FLAGS_MENU && win_list->window != Compositor.active_win && Gui.Window.IsVisible(win_list->window)) + @compositor_handle_window_input_events(win_list->window); + } + Compositor.max_z_index++; + win_list = win_list->next; + } +} + +U0 @compositor_reset_input_event_timestamps() +{ + Keyboard.last_key_tS = Keyboard.active_key_tS; +} + +Window* @compositor_create_window(I64 x, I64 y, I64 width, I64 height, + I64 flags = WIN_FLAGS_DEFAULT, + U8* title = NULL, Context2D* icon = NULL) +{ + Window* win = CAlloc(sizeof(Window)); + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + win->x = x; + win->y = y; + win->width = width; + win->height = height; + win->opacity = 255; + win->flags = flags; + win->icon = icon; + win->client = Fs; + win->signature = WIN_SIGNATURE; + win->widget = CAlloc(sizeof(@window_widgets_list)); + win->callback.close = &@gui_window_callback_close; + win->callback.maximize = &@gui_window_callback_maximize; + win->callback.minimize = &@gui_window_callback_minimize; + win->callback.repaint = NULL; + + if (title) + StrCpy(&win->title, title); + else + StrCpy(&win->title, "Untitled window"); + msg->client = Fs; + msg->type = CPZ_MSG_WIN_CREATE; + msg->payload = win; + Ipc.MsgSend(Compositor.task, msg); + while (!win->backing_store) { + Sleep(1); + }; + return win; +} + +U0 @compositor_destroy_window(Window* win) +{ + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + msg->client = Fs; + msg->type = CPZ_MSG_WIN_DESTROY; + msg->payload = win; + Ipc.MsgSend(Compositor.task, msg); +} + +U0 @compositor_hide_window(Window* win) +{ + @gui_window_hide(win); + Compositor.active_win = NULL; +} + +U0 @compositor_show_window(Window* win) { @gui_window_show(win); } + +Context2D* @compositor_win_select_backing_store(Window* win) +{ + I64 iterations = 0; + while (Compositor.in_resize && win == Compositor.active_win && (win->width != win->backing_store->width || win->height != win->backing_store->height) && iterations < 64) { + iterations++; + if (!(iterations % 8)) + @compositor_refresh(win); + Sleep(1); + } + return win->backing_store; +} + +U0 @compositor_blit_window_backing_stores() +{ + @compositor_windows_list* win_list; + win_list = Compositor.windows; + while (win_list) { + if (win_list->window) { + if (!(@gui_window_flag_is_set(win_list->window, WIN_FLAGS_MINIMIZED)) && !(@gui_window_flag_is_set(win_list->window, WIN_FLAGS_HIDDEN)) && win_list->window->signature == WIN_SIGNATURE) { + if (@compositor_win_select_backing_store(win_list->window) == win_list->window->resize_ctx) { + CopyRect2D(Compositor.ctx, win_list->window->x, win_list->window->y, + win_list->window->resize_ctx); + } else { + if (win_list->window->alpha) { + Compositor.blend_ctx->width = win_list->window->width; + Compositor.blend_ctx->height = win_list->window->height; + CopyRect2D(Compositor.blend_ctx, -win_list->window->x, + -win_list->window->y, Compositor.ctx); + BlendRect2D(win_list->window->backing_store, Compositor.blend_ctx); + CopyRect2D(Compositor.ctx, win_list->window->x, win_list->window->y, + Compositor.blend_ctx); + } else { + CopyRect2D(Compositor.ctx, win_list->window->x, win_list->window->y, + win_list->window->backing_store); + } + } + if (win_list->window == Compositor.active_win && !(@gui_window_flag_is_set(win_list->window, WIN_FLAGS_NOHILIGHT))) { + // Top border + Line2D(Compositor.ctx, win_list->window->x - 1, + win_list->window->y - 1, + win_list->window->x - 1 + win_list->window->width + 2, + win_list->window->y - 1, Compositor.theme.color.active_border); + // Left border + Line2D(Compositor.ctx, win_list->window->x - 1, + win_list->window->y - 1, win_list->window->x - 1, + win_list->window->y + win_list->window->height, + Compositor.theme.color.active_border); + // Bottom border + Line2D(Compositor.ctx, win_list->window->x - 1, + win_list->window->y + win_list->window->height, + win_list->window->x - 1 + win_list->window->width + 2, + win_list->window->y + win_list->window->height, + Compositor.theme.color.active_border); + // Right border + Line2D(Compositor.ctx, + win_list->window->x - 1 + win_list->window->width + 1, + win_list->window->y - 1, + win_list->window->x - 1 + win_list->window->width + 1, + win_list->window->y + win_list->window->height, + Compositor.theme.color.active_border); + // Bottom shadow + Line2D(Compositor.ctx, win_list->window->x + 1, + win_list->window->y + win_list->window->height + 1, + win_list->window->x + win_list->window->width + 2, + win_list->window->y + win_list->window->height + 1, + Compositor.theme.color.active_border); + // Right shadow + Line2D(Compositor.ctx, + win_list->window->x - 1 + win_list->window->width + 2, + win_list->window->y + 1, + win_list->window->x - 1 + win_list->window->width + 2, + win_list->window->y + win_list->window->height + 1, + Compositor.theme.color.active_border); + } + } + } + win_list = win_list->next; + } +} + +U0 @compositor_init() +{ + Compositor.blend_ctx = NewContext2D(Display.Width(), Display.Height()); + Compositor.ctx = NewContext2D(Display.Width(), Display.Height()); + Compositor.windows = CAlloc(sizeof(@compositor_windows_list)); + Compositor.global_input_event_listeners = CAlloc(sizeof(@compositor_windows_list)); + Compositor.bounds.x1 = -Display.Width(); + Compositor.bounds.y1 = 33; + Compositor.bounds.x2 = Display.Width(); + Compositor.bounds.y2 = Display.Height(); + Compositor.active_win = NULL; + Compositor.menubar.win = NULL; + Compositor.menubar.task = NULL; + Compositor.menubar.title = NULL; + Compositor.next_id = 0; + Compositor.session.user.uid = 1; + StrCpy(&Compositor.session.home, "/home/alec"); + StrCpy(&Compositor.session.hostname, "erythros"); + StrCpy(&Compositor.session.user.name, "alec"); + StrCpy(&Compositor.session.user.fullname, "Alec Murphy"); + @compositor_set_theme("Umami"); + @compositor_set_pointer(); +} + +U0 @compositor_task() +{ + Window* win; + Ipc.InitQueue(Fs); + Compositor.task = Fs; + System.Log(Fs, "Task running at 0x%08x", Fs); + + I64 total_mem; + I64 free_mem; + + while (1) { + // Process client IpcMessages. + @compositor_ipc_queue_process(); + + // Handle active window flags + @compositor_handle_active_window_flags(); + + // Handle window resize + @compositor_handle_window_resize(); + + // Handle window drag + @compositor_handle_window_drag(); + + // Handle window select + @compositor_handle_window_select(); + + // Handle global input events + @compositor_handle_global_input_events(); + + // Handle windows input events + @compositor_handle_windows_input_events(); + + // Reset input event timestamps + @compositor_reset_input_event_timestamps(); + + // Blit window backing stores + @compositor_blit_window_backing_stores(); + + // Draw Mouse Pointer (if no mouse integration). + // if (!Mouse.Update || Mouse.integration_type == MI_QEMU) + Blot2D(Compositor.ctx, Mouse.X(), Mouse.Y(), Compositor.pointer); + + // Debug stuff + I64 debug_row; + + if (1 == 2) { + debug_row = 2; + + total_mem = sys_code_bp->alloced_u8s; + if (sys_data_bp) { + total_mem += sys_data_bp->alloced_u8s; + } + free_mem = sys_code_bp->alloced_u8s - sys_code_bp->used_u8s; + if (sys_data_bp) { + free_mem += sys_data_bp->alloced_u8s - sys_data_bp->used_u8s; + } + + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "SysTimerRead : 0x%08x", SysTimerRead); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Total Memory : 0x%08x", total_mem); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Free Memory : 0x%08x", free_mem); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Mouse.X() : %d", Mouse.X()); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Mouse.Y() : %d", Mouse.Y()); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Mouse.Z() : %d", Mouse.z); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Compositor.mouse.z : %d", Compositor.mouse.z); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Mouse.left() : %d", Compositor.mouse.left > 0); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Mouse.right() : %d", Compositor.mouse.right > 0); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "Active window: 0x%08x", Compositor.active_win); + if (Compositor.active_win) { + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + "[+] x : %d", Compositor.active_win->x); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + " y : %d", Compositor.active_win->y); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + " width : %d", Compositor.active_win->width); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + " height : %d", Compositor.active_win->height); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + " title : %s", Compositor.active_win->title); + ConsolePrint2D(Compositor.ctx, 08 * 03, 16 * debug_row++, , , + " flags : %016b", Compositor.active_win->flags); + } + } + + // Flip off-screen context to framebuffer. + Graphics2D.Flip(Compositor.ctx); + + // Update Compositor mouse button state + Compositor.mouse.x = Mouse.x; + Compositor.mouse.y = Mouse.y; + if (Compositor.mouse.delta_z != Mouse.z) + Compositor.mouse.z = (Mouse.z - Compositor.mouse.delta_z) * T(Mouse.natural_scroll, -Mouse.wheel_sensitivity, Mouse.wheel_sensitivity); + Compositor.mouse.delta_z = Mouse.z; + Compositor.mouse.left = Mouse.left; + Compositor.mouse.right = Mouse.right; + + if (Display.Update) + Display.Update(); + Sleep(1); + } +} + +Gui.Window.Refresh = &@compositor_refresh; +Gui.Window.SetFocus = &@compositor_set_active_window; +Gui.Window.SetZIndex = &@compositor_set_z_index_send_msg; + +Compositor.GetWindowByTitle = &@compositor_get_window_by_title; +Compositor.GetWindowByZIndex = &@compositor_get_window_by_z_index; +Compositor.Init = &@compositor_init; +Compositor.CreateWindow = &@compositor_create_window; +Compositor.DestroyWindow = &@compositor_destroy_window; +Compositor.HideWindow = &@compositor_hide_window; +Compositor.RegisterForGlobalInputEvents = &@compositor_register_global_input_event_listener; +Compositor.ShowWindow = &@compositor_show_window; +Compositor.Task = &@compositor_task; +Compositor.UnregisterForGlobalInputEvents = &@compositor_unregister_global_input_event_listener; +Compositor.SetWallpaper = &@compositor_set_wallpaper_send_msg; + +"compositor "; \ No newline at end of file diff --git a/System/Core/FileSystem.HC b/System/Core/FileSystem.HC new file mode 100644 index 0000000..c9b8dc5 --- /dev/null +++ b/System/Core/FileSystem.HC @@ -0,0 +1,3 @@ + + +"filesystem "; \ No newline at end of file diff --git a/System/Core/Menu.HC b/System/Core/Menu.HC new file mode 100644 index 0000000..8983c9a --- /dev/null +++ b/System/Core/Menu.HC @@ -0,0 +1,67 @@ +// Core component for Menu functions + +#define MENU_ITEM_MIN_HEIGHT 24 +#define MENU_ITEM_MIN_WIDTH 256 + +I64 @menu_get_items_count(Window* win) +{ + I64 count = 0; + @window_widgets_list* wl = win->widget; + while (wl) { + if (wl->widget) { + if (wl->widget->type == WIDGET_TYPE_MENU_ITEM) + count++; + } + wl = wl->next; + } + return count; +} + +MenuItemWidget* @menu_add_item(Window* win, U8* text, Context2D* icon, + U64 callback, U8* path = NULL, + Window* submenu = NULL) +{ + I64 items_count = @menu_get_items_count(win); + win->height = 8; + win->height += MENU_ITEM_MIN_HEIGHT * (items_count + 1); + MenuItemWidget* item = Gui.CreateWidget( + win, WIDGET_TYPE_MENU_ITEM, 0, MENU_ITEM_MIN_HEIGHT * items_count, + MENU_ITEM_MIN_WIDTH, MENU_ITEM_MIN_HEIGHT); + Gui.Widget.SetText(item, text); + if (icon) + item->icon = icon; + if (path) + item->path = StrNew(path); + if (submenu) + item->submenu = submenu; + if (callback) + Gui.Widget.SetCallback(item, "clicked", callback); + return item; +} + +Window* @menu_new(U8* title = NULL) +{ + Window* menu = Compositor.CreateWindow( + 0, 0, MENU_ITEM_MIN_WIDTH, MENU_ITEM_MIN_HEIGHT, + WIN_FLAGS_NOHILIGHT | WIN_FLAGS_SKIP | WIN_FLAGS_MENU); + Gui.Window.Hide(menu); + if (title) + Gui.Window.SetTitle(menu, title); + else + Gui.Window.SetTitle(menu, ""); + return menu; +} + +class @menu +{ + MenuItemWidget* (*AddItem)(Window* win, U8* text, Context2D* icon, + U64 callback, U8* path = NULL, + Window* submenu = NULL); + Window* (*New)(U8* title); +}; + +@menu Menu; +Menu.AddItem = &@menu_add_item; +Menu.New = &@menu_new; + +"menu "; \ No newline at end of file diff --git a/System/Core/MessageBox.HC b/System/Core/MessageBox.HC new file mode 100644 index 0000000..8f8f1d8 --- /dev/null +++ b/System/Core/MessageBox.HC @@ -0,0 +1,108 @@ +// Core component for MessageBox functions + +Context2D* MESSAGEBOX_ICON_ERR = Image.FileToContext2D( + "/Media/Themes/Umami/Icon/status/messagebox_critical.png"); +Context2D* MESSAGEBOX_ICON_INFO = Image.FileToContext2D("/Media/Themes/Umami/Icon/status/messagebox_info.png"); +Context2D* MESSAGEBOX_ICON_WARN = Image.FileToContext2D( + "/Media/Themes/Umami/Icon/status/messagebox_warning.png"); + +Context2D* MESSAGEBOX_WIN_ICON_ERR = Image.FileToContext2D( + "/Media/Themes/Umami/Icon/status/messagebox_critical_16x16.png"); +Context2D* MESSAGEBOX_WIN_ICON_INFO = Image.FileToContext2D( + "/Media/Themes/Umami/Icon/status/messagebox_info_16x16.png"); +Context2D* MESSAGEBOX_WIN_ICON_WARN = Image.FileToContext2D( + "/Media/Themes/Umami/Icon/status/messagebox_warning_16x16.png"); + +#define MESSAGEBOX_TYPE_ALERT 0 +#define MESSAGEBOX_TYPE_ERROR 1 +#define MESSAGEBOX_TYPE_INFO 2 + +U0 @messagebox_close_window(Window* window) +{ + Compositor.DestroyWindow(window); +} +U0 @messagebox_close_widget(Widget* widget) +{ + Compositor.DestroyWindow(widget->parent_win); +} + +U0 @messagebox_msg(U8* str, I64 type, Context2D* win_icon, Context2D* icon, + U8* options = NULL, U64 callback = NULL) +{ + U64 flags = WIN_FLAGS_MOVABLE | WIN_FLAGS_ICON | WIN_FLAGS_TITLE_BAR | WIN_FLAGS_CLOSE_BUTTON; + Window* win = Compositor.CreateWindow(0, 0, 320, 192, flags); + Gui.Window.SetIcon(win, win_icon); + switch (type) { + case MESSAGEBOX_TYPE_ALERT: + Gui.Window.SetTitle(win, "Alert"); + break; + case MESSAGEBOX_TYPE_ERROR: + Gui.Window.SetTitle(win, "Error"); + break; + case MESSAGEBOX_TYPE_INFO: + Gui.Window.SetTitle(win, "Info"); + break; + } + Context2DWidget* ctx_icon = Gui.CreateWidget(win, WIDGET_TYPE_CONTEXT2D, 8, 16, 24, 24); + ctx_icon->ctx = icon; + TextLabelWidget* lbl_text = Gui.CreateWidget(win, WIDGET_TYPE_LABEL, 40, 16, 192, 96); + ButtonWidget* btn_ok = NULL; + ButtonWidget* btn_cancel = NULL; + + Gui.Widget.SetText(lbl_text, str); + Gui.Window.SetCallback(win, "close", callback); + + if (options) { + btn_ok = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, (win->width / 2) - 80, + win->height - 60, 64, 24); + btn_cancel = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, (win->width / 2) + 16, + win->height - 60, 64, 24); + Gui.Widget.SetText(btn_ok, " OK "); + Gui.Widget.SetText(btn_cancel, " Cancel "); + Gui.Widget.SetCallback(btn_ok, "clicked", callback); + Gui.Widget.SetCallback(btn_cancel, "clicked", callback); + btn_ok->tag = TRUE; + btn_cancel->tag = FALSE; + } else { + btn_ok = Gui.CreateWidget(win, WIDGET_TYPE_BUTTON, (win->width / 2) - 32, + win->height - 60, 64, 24); + Gui.Widget.SetText(btn_ok, " OK "); + Gui.Widget.SetCallback(btn_ok, "clicked", callback); + btn_ok->tag = TRUE; + } + + Gui.Window.Center(win); + Gui.Window.SetFocus(win); +} + +U0 @messagebox_alert(U8* str, U8* options = NULL, U64 callback = NULL) +{ + @messagebox_msg(str, MESSAGEBOX_TYPE_ALERT, MESSAGEBOX_WIN_ICON_WARN, + MESSAGEBOX_ICON_WARN, options, callback); +} + +U0 @messagebox_error(U8* str, U8* options = NULL, U64 callback = NULL) +{ + @messagebox_msg(str, MESSAGEBOX_TYPE_ERROR, MESSAGEBOX_WIN_ICON_ERR, + MESSAGEBOX_ICON_ERR, options, callback); +} + +U0 @messagebox_info(U8* str, U8* options = NULL, U64 callback = NULL) +{ + @messagebox_msg(str, MESSAGEBOX_TYPE_INFO, MESSAGEBOX_WIN_ICON_INFO, + MESSAGEBOX_ICON_INFO, options, callback); +} + +class @messagebox +{ + U0* (*Alert)(U8* str, U8* options = NULL, U64 callback = NULL); + U0* (*Error)(U8* str, U8* options = NULL, U64 callback = NULL); + U0* (*Info)(U8* str, U8* options = NULL, U64 callback = NULL); +}; + +@messagebox MessageBox; +MessageBox.Alert = &@messagebox_alert; +MessageBox.Error = &@messagebox_error; +MessageBox.Info = &@messagebox_info; + +"messagebox "; \ No newline at end of file diff --git a/System/Core/Scheduler.HC b/System/Core/Scheduler.HC new file mode 100644 index 0000000..6415f07 --- /dev/null +++ b/System/Core/Scheduler.HC @@ -0,0 +1,79 @@ +/* clang-format off */ + +U0 @scheduler_restore_page_table(U64 i) { + U64 reg R15 _pt = i.u32[1]; + if (_pt) { // Use CTask's page table entries + asm { + MOV RAX, R15 + MOV_CR3_RAX + } + } else { // Use identity mapping + asm { + MOV EAX, [MEM_PML4] + MOV_CR3_RAX + } + } +} + +asm { +ERYTHROS_TASK_CONTEXT_RESTORE_START:: + XOR RAX,RAX + INC U64 GS:CCPU.swap_cnter[RAX] + MOV RSI,FS:CTask.addr[RAX] + BT U32 CTask.rflags[RSI],RFLAGf_INT + JNC @@05 + BTS U32 GS:CCPU.cpu_flags[RAX],CPUf_RAN_A_TASK +@@05: BT U64 CTask.task_flags[RSI],TASKf_DISABLE_BPTS + JC @@15 + MOV RDX,U64 CTask.bpt_lst[RSI] +@@10: TEST RDX,RDX + JZ @@15 + MOV RDI,U64 CBpt.addr[RDX] + MOV U8 [RDI],OC_BPT + MOV RDX,U64 CBpt.next[RDX] + JMP @@10 + +@@15: INC U64 CTask.swap_cnter[RSI] + + MOV RAX, U64 CTask.user_data[RSI] + PUSH RAX + CALL I32 &@scheduler_restore_page_table + + MOV RAX,U64 CTask.fpu_mmx[RSI] + FXRSTOR U64 [RAX] + + MOV RAX,RSP + LEA RSP,U64 CTask.rcx[RSI] + POP RCX + POP RDX + POP RBX + POP RBP + POP RDI + POP R8 + POP R9 + POP R10 + POP R11 + POP R12 + POP R13 + POP R14 + POP R15 + MOV RSP,RAX + + MOV RAX,U64 CTask.rax[RSI] + PUSH CGDT.ds + PUSH U64 CTask.rsp[RSI] + PUSH U64 CTask.rflags[RSI] + PUSH CGDT.cs64 + PUSH U64 CTask.rip[RSI] + MOV RSI,U64 CTask.rsi[RSI] + IRET + ERYTHROS_TASK_CONTEXT_RESTORE_END:: + NOP + //************************************ +} + +/* clang-format on */ + +Function.Patch(_TASK_CONTEXT_RESTORE, ERYTHROS_TASK_CONTEXT_RESTORE_START); + +"scheduler "; \ No newline at end of file diff --git a/System/Core/Shell.HC b/System/Core/Shell.HC new file mode 100644 index 0000000..90b2a95 --- /dev/null +++ b/System/Core/Shell.HC @@ -0,0 +1,282 @@ +#define SHELL_OPTS_ERR_INVALID_OPT -1 +#define SHELL_OPTS_ERR_EXTRA_OPD -2 + +U0 @shell_free_args(I64 argc, U8** argv) +{ + I64 i; + for (i = 0; i < argc; i++) + Free(argv[i]); + if (argv) + Free(argv); +} + +U8** @shell_parse_args(@shell* sh, U8* str, + I64* argc) +{ // Return argc, argv from str. + Bool quoted = FALSE; + I64 _argc = 0; + U8** _argv = NULL; + U8** _tmp = CAlloc(sizeof(U64) * StrLen(str)); + I64 i = 0; + I64 s = 0; + I64 len; + while (i < StrLen(str) + 1) { + switch (str[i]) { + case 0: + case ' ': + if (!quoted) { + len = (str + i) - (str + s - 1); + if (str[i - 1] == '"') + len--; + if (len - 1) { + _tmp[_argc] = CAlloc(len); + MemCpy(_tmp[_argc], str + s, len - 1); + _argc++; + } + s = i + 1; + } + break; + case '"': + quoted = !quoted; + if (quoted) + s = i + 1; + break; + default: + break; + } + i++; + } + *argc = _argc; + _argv = CAlloc(sizeof(U64) * _argc); + MemCpy(_argv, _tmp, sizeof(U64) * _argc); + Free(_tmp); + return _argv; +} + +I64 @shell_parse_opts(@shell* sh, U8* op_lst, I64 argc, U8** argv, I64* flags, + U64* op_err, Bool ignore_extra_opd = FALSE) +{ + I64 i, j; + U8 op_chr[2]; + op_chr[1] = 0; + for (i = 1; i < argc; i++) { + if (argv[i][0] == '-') { + for (j = 1; j < StrLen(argv[i]); j++) { + op_chr[0] = argv[i][j]; + if (StrFind(&op_chr, op_lst)) + *flags |= 1 << (StrFind(&op_chr, op_lst) - op_lst); + else { + *op_err = StrNew(&op_chr); + return SHELL_OPTS_ERR_INVALID_OPT; + } + } + } else { + if (!ignore_extra_opd) { + *op_err = StrNew(argv[i]); + return SHELL_OPTS_ERR_EXTRA_OPD; + } + } + } + return 0; +} +U8* @shell_expand_relative_path(@shell* sh, U8* path) +{ + if (!path || !sh) + return NULL; + if (StrLen(path) < 1) + return NULL; + switch (path[0]) { + case '/': + return StrNew(path); + break; + default: + U8* abs_path = CAlloc(StrLen(path) + StrLen(&sh->cwd) + 4); + StrPrint(abs_path, "%s/%s", &sh->cwd, path); + return abs_path; + break; + } +} + +U8* @shell_get_env_var(@shell* sh, U8* key) +{ + @shell_env_var* var = sh->env->next; + while (var) { + if (StrLen(&var->key) && StrLen(&var->value)) + if (!StrCmp(&var->key, key)) + return &var->value; + var = var->next; + } + return ""; +} + +U0 @shell_set_env_var(@shell* sh, U8* key, U8* value) +{ + if (!sh || !key || !value) + return; + @shell_env_var* var = sh->env->next; + while (var->next) { + if (!StrCmp(&var->key, key)) { + StrCpy(&var->value, value); + return; + } + var = var->next; + } + @shell_env_var* new = CAlloc(sizeof(@shell_env_var)); + StrCpy(&new->key, key); + StrCpy(&new->value, value); + new->prev = var; + var->next = new; +} + +U0 @shell_unset_env_var(@shell* sh, U8* key) +{ + @shell_env_var* var = sh->env->next; + @shell_env_var* prev = NULL; + @shell_env_var* next = NULL; + while (var) { + if (!StrCmp(&var->key, key)) { + prev = var->prev; + next = var->next; + if (prev) + prev->next = next; + if (next) + next->prev = prev; + Free(var); + return; + } + var = var->next; + } +} + +U0 @shell_history_append(@shell* sh, U8* str) +{ + if (!sh || !str) + return; + if (!StrCmp(str, "")) + return; + I64 i; + sh->history.entries[sh->history.pos] = StrNew(str); + sh->history.pos++; + if (sh->history.pos > SHELL_HISTORY_LIMIT - 1) { + Free(sh->history.entries[0]); + for (i = 0; i < SHELL_HISTORY_LIMIT; i++) + sh->history.entries[i] = sh->history.entries[i + 1]; + sh->history.pos--; + } +} + +U0 @shell_process_args(@shell* sh, I64 argc, U8** argv) +{ + if (argc < 1 || !argv) + return; + I64 i; + U8 buf[256]; + for (i = 0; i < argc; i++) { + if (argv[i][0] == '\d') { + switch (argv[i][1]) { + case '?': + Free(argv[i]); + StrPrint(&buf, "%d", sh->answer); + argv[i] = StrNew(&buf); + break; + case '0': + Free(argv[i]); + argv[i] = StrNew("esh"); + break; + default: + Free(argv[i]); + argv[i] = StrNew(@shell_get_env_var(sh, argv[i] + 1)); + break; + } + } + if (!StrCmp(argv[i], "~")) { + Free(argv[i]); + argv[i] = StrNew(&sh->session->home); + } + } +} + +U0 @shell_update_prompts(@shell* sh) +{ + U8 buf[512]; + U8 buf2[512]; + StrCpy(&buf, &sh->cwd); + StrPrint(buf2, "/home/%s", &sh->session->user.name); + if (!StrCmp(&buf, &buf2)) + StrCpy(&buf, "~"); + else + StrCpy(&buf, StrLastOcc(&buf, "/") + 1); + StrPrint(&sh->PS1, "[%s@%s %s]\d ", &sh->session->user.name, + &sh->session->hostname, &buf); +} + +I64 @shell_input_loop(@shell* sh) +{ + CHashFun* cmd; + I64 argc; + U8** argv; + U8 buf[4096]; + Bool exit = FALSE; + I64 i; + I64 (*@shell_exec)(@shell* sh, I64 argc, U8** argv); + + while (!exit) { + + @shell_update_prompts(sh); + Stdio.WriteLine(sh, &sh->PS1); + Stdio.ReadLine(sh, &buf); + + @shell_history_append(sh, &buf); + argv = @shell_parse_args(sh, &buf, &argc); + + if (argc) { + if (!StrCmp(argv[0], "exit")) { + exit = TRUE; + goto @shell_exit; + } + @shell_process_args(sh, argc, argv); + StrPrint(&buf, "@shell_cmd_%s", argv[0]); + cmd = HashFind(&buf, adam_task->hash_table, HTT_FUN); + if (cmd) { + @shell_exec = cmd->exe_addr; + sh->answer = @shell_exec(sh, argc, argv); + sh->break = FALSE; + FifoU8Flush(sh->input); + } else { + StrPrint(&buf, "%s: command not found\n", argv[0]); + Stdio.WriteLine(sh, &buf); + sh->answer = 0; + } + } + + @shell_exit : @shell_free_args(argc, argv); + } + return 0; +} + +U0 @shell_instance(@shell* sh) +{ + @shell_input_loop(sh); + sh->exit = TRUE; +} + +U0 @shell_init(@shell* sh) +{ + sh->env = CAlloc(sizeof(@shell_env_var)); + sh->history.limit = SHELL_HISTORY_LIMIT; + sh->history.pos = 0; + sh->history.entries = CAlloc(sizeof(U64) * SHELL_HISTORY_LIMIT); + sh->input = FifoU8New(SHELL_INPUT_FIFO_SIZE); + sh->task = Spawn(&@shell_instance, sh); +} + +@shell* @shell_new(Bool headless = FALSE) +{ + @shell* sh = CAlloc(sizeof(@shell)); + if (!headless) + @shell_init(sh); + StrCpy(&sh->cwd, &Compositor.session.home); + return sh; +} + +"shell "; \ No newline at end of file diff --git a/System/Core/ShellCommands.HC b/System/Core/ShellCommands.HC new file mode 100644 index 0000000..c5d458a --- /dev/null +++ b/System/Core/ShellCommands.HC @@ -0,0 +1,13 @@ +extern I64 @systemstarter_open(@shell* sh, I64 argc, U8** argv); + +CDirEntry* sc_de = FilesFind("M:/System/Shell/Commands/*.HC"); +CDirEntry* sc_de2 = sc_de; +while (sc_de2) { + if (!(!StrCmp(sc_de2->name, ".") && !StrCmp(sc_de2->name, ".."))) { + ExeDoc(DocRead(sc_de2->full_name)); + } + sc_de2 = sc_de2->next; +} +DirTreeDel(sc_de); + +"shellcommands "; \ No newline at end of file diff --git a/System/Core/SystemStarter.HC b/System/Core/SystemStarter.HC new file mode 100644 index 0000000..bf7fa4a --- /dev/null +++ b/System/Core/SystemStarter.HC @@ -0,0 +1,214 @@ +#define SYSSTART_MSG_NULL 0 +#define SYSSTART_MSG_SPAWN 1 +#define SYSSTART_MSG_KILL 2 + +class @systemtask +{ + U8* path; + U8* name; +}; + +class @systemstarter +{ + CTask* task; + U0 (*CreateTask)(U8* path, U8* name); + U0 (*Init)(); + U0 (*Spawn)(U8* path, U8* name); + U0 (*Task)(); +}; + +@systemstarter SystemStarter; + +U32 @systemstarter_ext(U8* path) +{ + U32 res = NULL; + U8* ext = FileSystem.GetFileExtension(path); + MemCpy(&res, ext, StrLen(ext)); + return res; +} + +U0 @systemstarter_init() { } + +I64 @systemstarter_open(@shell* sh, I64 argc, U8** argv) +{ + U8 buf[512]; + U8* path = @shell_expand_relative_path(sh, argv[1]); + if (!FileSystem.PathExists(path)) { + Stdio.WriteLine(sh, "error: path does not exist: "); + Stdio.WriteLine(sh, path); + Stdio.WriteLine(sh, "\n"); + Free(path); + return 1; + } + + switch (@systemstarter_ext(path)) { + case 'app': + StrCpy(&buf, ""); + String.Append(&buf, "I64 @exe_doc_buf_size = NULL; "); + String.Append(&buf, + "U8 *@exe_doc_buf = FileSystem.ReadFile(\"%s/Run.HC\", " + "&@exe_doc_buf_size); ", + path); + String.Append(&buf, "ExePutS(@exe_doc_buf); "); + String.Append(&buf, "Free(@exe_doc_buf); "); + // System.Log(Fs, &buf); + CTask* task = User; + TaskExe(task, NULL, "Raw(ON);\n", 0); + TaskExe(task, NULL, &buf, 0); + return 0; + break; + default: + Stdio.WriteLine(sh, "error: unknown or unsupported file type.\n"); + // Free(path); + return 1; + break; + }; +} + +U0 @systemstarter_spawn(U8* path, U8* name) +{ + CTask* task = Spawn(&UserCmdLine); + Sleep(Rand * 100); + U8 change_path_str[512]; + StrPrint(task->task_name, name); + StrPrint(change_path_str, "Cd(\"%s\");\n", path); + TaskExe(task, NULL, "Raw(ON);\n", 0); + TaskExe(task, NULL, change_path_str, 0); + TaskExe(task, NULL, "ExeFile(\"Run.HC\");\n", 0); + Sleep(Rand * 100); +} + +U0 @systemstarter_load_applets() +{ + U8 applet_name[512]; + CDirEntry* de = FilesFind("M:/Applets/*.applet"); + CDirEntry* tmpde = de; + while (tmpde) { + StrCpy(&applet_name, StrLastOcc(tmpde->full_name, "/") + 1); + *(StrFirstOcc(&applet_name, ".")) = NULL; + SystemStarter.Spawn(tmpde->full_name, &applet_name); + tmpde = tmpde->next; + } + DirTreeDel(de); +} + +U0 @systemstarter_play_user_startup_sound() +{ + U8 path[512]; + StrPrint(&path, "/home/%s/.sounds/startup.wav", + &Compositor.session.user.name); + U8** argv = CAlloc(sizeof(U64) * 2); + argv[0] = "aplay"; + argv[1] = &path; + @shell* sh = @shell_new(TRUE); + sh->session = &Compositor.session; + //@shell_cmd_aplay(sh, 2, argv); + Free(sh); +} + +U0 @systemstarter_set_user_wallpaper() +{ + U8 path[512]; + StrPrint(&path, "/home/%s/.wallpaper/wallpaper.png", + &Compositor.session.user.name); + U8** argv = CAlloc(sizeof(U64) * 2); + argv[0] = "wpset"; + argv[1] = &path; + //@shell_cmd_wpset(NULL, 2, argv); +} + +U0 @systemstarter_user_startup() +{ + // Set User wallpaper + @systemstarter_set_user_wallpaper(); + + // Play User startup sound + @systemstarter_play_user_startup_sound(); +} + +U0 @systemstarter_startup() +{ + + // Set user-specific startup preferences + Spawn(&@systemstarter_user_startup); + + // Initialize Clipboard + Clipboard.Init(); + + // Spawn Clipboard Task + Spawn(Clipboard.Task, , "Clipboard", T(mp_cnt > 3, 2, 1)); + + // Initialize SystemTray + SystemTray.Init(); + + // Spawn SystemTray Task + Spawn(SystemTray.Task, , "SystemTray", T(mp_cnt > 3, 2, 1)); + + SystemStarter.Spawn("M:/Applications/OS/Wallpaper.app", "Wallpaper"); + SystemStarter.Spawn("M:/Applications/OS/MenuBar.app", "MenuBar"); + SystemStarter.Spawn("M:/Applications/OS/TaskSwitcher.app", + "TaskSwitcher"); + + // Load SystemTray Applets + @systemstarter_load_applets; +} + +U0 @systemstarter_ipc_queue_process() +{ + IpcMessage* msg; + @systemtask* st = NULL; + msg = Ipc.MsgRecv(); + if (msg) { + switch (msg->type) { + case SYSSTART_MSG_SPAWN: + if (msg->payload) { + st = msg->payload; + if (st->path && st->name) { + SystemStarter.Spawn(st->path, st->name); + System.Log(Fs, "Received message ← CreateTask (%s)", st->name); + Free(st->name); + Free(st->path); + } + Free(st); + } + break; + case SYSSTART_MSG_KILL: + break; + default: + break; + } + Free(msg); + } +} + +U0 @systemstarter_task() +{ + Ipc.InitQueue(Fs); + System.Log(Fs, "Task running at 0x%08x", Fs); + SystemStarter.task = Fs; + Spawn(&@systemstarter_startup, , , T(mp_cnt, 1, 0)); + + while (1) { + @systemstarter_ipc_queue_process; + Sleep(1); + } +} + +U0 @systemstarter_create_task(U8* path, U8* name) +{ + @systemtask* st = CAlloc(sizeof(@systemtask)); + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + st->path = StrNew(path); + st->name = StrNew(name); + msg->client = NULL; + msg->type = SYSSTART_MSG_SPAWN; + msg->payload = st; + Ipc.MsgSend(SystemStarter.task, msg); +} + +SystemStarter.CreateTask = &@systemstarter_create_task; +SystemStarter.Init = &@systemstarter_init; +SystemStarter.Spawn = &@systemstarter_spawn; +SystemStarter.Task = &@systemstarter_task; + +"systemstarter "; diff --git a/System/Core/SystemTray.HC b/System/Core/SystemTray.HC new file mode 100644 index 0000000..db7c7e6 --- /dev/null +++ b/System/Core/SystemTray.HC @@ -0,0 +1,139 @@ +#define SYSTRAY_MSG_NULL 0x0 +#define SYSTRAY_MSG_REGISTER 0x1 +#define SYSTRAY_MSG_UNREGISTER 0x2 + +class @systemtray +{ + CTask* task; + @window_widgets_list* item; + U0 (*Init)(); + U0 (*Task)(); + Context2DWidget* (*RegisterItem)(); + U0 (*UnregisterItem)(Widget* widget); + U0 (*SetIcon)(Context2DWidget* widget, U8* path); +}; + +@systemtray SystemTray; + +U0 @systemtray_register_item(I64 addr) +{ + Context2DWidget* item = Gui.CreateWidget( + Compositor.menubar.win, WIDGET_TYPE_CONTEXT2D, -24, 0, 24, 24); + item->ctx = NewContext2D(item->width, item->height); + Fill2D(item->ctx, Color(0, 0, 0, 0)); + MemSetI64(addr, item, 1); +} + +U0 @systemtray_unregister_item(Widget* item) +{ + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + msg->client = NULL; + msg->type = CPZ_MSG_WIN_WIDGET_DESTROY; + msg->payload = item; + System.Log(Fs, "Sent message → WidgetDestroy"); + Ipc.MsgSend(Compositor.menubar.task, msg); +} + +U0 @systemtray_reindex_items() +{ + I64 x = Display.Width() - 100; + @window_widgets_list* item = Compositor.menubar.win->widget; + while (item->next) + item = item->next; + while (item->widget->type == WIDGET_TYPE_CONTEXT2D) { + x -= item->widget->width + 4; + item->widget->x = x; + item = item->prev; + } + Gui.Window.Refresh(Compositor.menubar.win); +} + +U0 @systemtray_ipc_queue_process() +{ + IpcMessage* msg; + msg = Ipc.MsgRecv(); + if (msg) { + switch (msg->type) { + case SYSTRAY_MSG_REGISTER: + @systemtray_register_item(msg->payload); + @systemtray_reindex_items; + break; + case SYSTRAY_MSG_UNREGISTER: + @systemtray_unregister_item(msg->payload); + @systemtray_reindex_items; + break; + default: + break; + } + Free(msg); + } +} + +U0 @systemtray_init() { } + +U0 @systemtray_task() +{ + Ipc.InitQueue(Fs); + SystemTray.task = Fs; + System.Log(Fs, "Task running at 0x%08x", Fs); + while (!Compositor.menubar.win) // Wait for instance + Sleep(1); + while (1) { + @systemtray_ipc_queue_process(); + Sleep(1); + } +} + +Context2DWidget* @systemtray_client_register_item() +{ + Context2DWidget* item = NULL; + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + msg->client = NULL; + msg->type = SYSTRAY_MSG_REGISTER; + msg->payload = &item; + System.Log(Fs, "Sent message → SystrayRegisterItem"); + Ipc.MsgSend(SystemTray.task, msg); + while (!item) + Sleep(1); + return item; +} + +U0 @systemtray_client_unregister_item(Widget* item) +{ + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + msg->client = NULL; + msg->type = SYSTRAY_MSG_UNREGISTER; + msg->payload = item; + System.Log(Fs, "Sent message → SystrayUnRegisterItem"); + Ipc.MsgSend(SystemTray.task, msg); + while (item->type) + Sleep(1); + @systemtray_reindex_items; +} + +U0 @systemtray_set_icon(Context2DWidget* widget, U8* path) +{ + if (!widget || !path) + return; + U8 full_path[512]; + if (2 == 3) // FIXME + StrCpy(&full_path, path); + else { + StrCpy(&full_path, "M:/Media/Themes/Umami/Icon/"); + String.Append(&full_path, path); + } + if (!FileFind(&full_path)) + return; + Context2D* icon = Image.FileToContext2D(&full_path); + CopyRect2D(widget->ctx, 0, 0, icon); + DelContext2D(icon); + Gui.Window.Refresh(Compositor.menubar.win); +} + +SystemTray.Init = &@systemtray_init; +SystemTray.RegisterItem = &@systemtray_client_register_item; +SystemTray.SetIcon = &@systemtray_set_icon; +SystemTray.Task = &@systemtray_task; +SystemTray.UnregisterItem = &@systemtray_client_unregister_item; + +"systemtray "; \ No newline at end of file diff --git a/System/Drivers/AC97.HC b/System/Drivers/AC97.HC new file mode 100644 index 0000000..77732d2 --- /dev/null +++ b/System/Drivers/AC97.HC @@ -0,0 +1,153 @@ +#define INT_LAST_VALID_ENTRY 1 << 2 +#define INT_IOC 1 << 3 +#define INT_FIFO_ERR 1 << 4 + +#define BDL_BUF_SIZE 2044 +#define PCM_BUF_SIZE 2048 +#define MAX_BDLS 32 + +#define PCM_IN 0 +#define PCM_OUT 1 +#define MIC_IN 2 + +// Native Audio Mixer registers (all U16) + +#define RESET 0x00 // Reset Register +#define MASTER_VOL 0x02 // Set Master Output Volume +#define MIC_VOL 0x0E // Set Microphone Volume +#define PCM_VOL 0x18 // Set Output Volume of PCM patterns +#define REC_SLC 0x1A // Select Input Device +#define REC_GAIN 0x1C // Set Input Gain +#define MIC_GAIN 0x1E // Set Gain of Microphone +#define EXT_ID 0x28 // Supported extended functions +#define EXT_CTRL 0x2A // Enabling extended functions +#define EXT_FRONT_RATE 0x2C // Sample rate of front speaker + +// Native Audio Bus Master registers + +#define PCM_INPUT_REG_BOX \ + 0x00 // NABM register box for PCM IN (sizeof NABM register box) +#define PCM_OUTPUT_REG_BOX \ + 0x10 // NABM register box for PCM OUT (sizeof NABM register box) +#define MIC_INPUT_REG_BOX \ + 0x20 // NABM register box for Microphone (sizeof NABM register box) +#define GLOBAL_CTL 0x2C // Global Control Register (U32) +#define GLOBAL_STS 0x30 // Global Status Register (U32) + +// NABM register box registers + +#define BUFFER_DSC_ADDR 0x00 // Physical Address of Buffer Descriptor List (U32) +#define CUR_ENTRY_VAL \ + 0x04 // Number of Actual Processed Buffer Descriptor Entry (U8) +#define LAST_VALID_ENTRY 0x05 // Number of all Descriptor Entries (U8) +#define TRANSFER_STS 0x06 // Status of Transferring Data (U16) +#define CUR_IDX_PROC_SAMPLES \ + 0x08 // Number of Transferred Samples in Actual Processed Entry (U16) +#define PRCSD_ENTRY 0x0A // Number of Actual Processed Buffer Entry (U8) +#define BUFFER_CNT \ + 0x0B // Most Important Register for controlling Transfers (U8) + +class @ac97_bdl_entry +{ + U32 addr; + U16 length; // length - 1 + U16 flags; +}; + +class @ac97_bdl +{ + @ac97_bdl_entry entries[32]; +}; + +class @ac97 +{ + @pci_info pci; + @ac97_bdl* bdl[3]; + U16 nam; + U16 nabm; +}; + +@ac97 AC97; + +U0 @ac97_fill_buffer() +{ + I64 idx = InU8(AC97.nabm + PCM_OUTPUT_REG_BOX + LAST_VALID_ENTRY); + U32* buf = AC97.bdl[PCM_OUT]->entries[idx].addr; + @audio_mix_output(buf, BDL_BUF_SIZE); + OutU8(AC97.nabm + PCM_OUTPUT_REG_BOX + LAST_VALID_ENTRY, ++idx); +} + +U0 @ac97_int_handler() +{ + U16 status = InU16(AC97.nabm + PCM_OUTPUT_REG_BOX + TRANSFER_STS); + if (status & INT_IOC) { + @ac97_fill_buffer; + OutU16(AC97.nabm + PCM_OUTPUT_REG_BOX + TRANSFER_STS, 0x1C); + } +} + +I64 @ac97_init() +{ + I64 i; + I64 j; + // Scan for device + j = PCIClassFind(0x040100, 0); + if (j < 0) { + device_not_found: + AdamLog("\n[AC'97] Device not found\n"); + return -1; + } + @get_pci_info(j, &AC97.pci); + + if (AC97.pci.vendor_id != 0x8086 || AC97.pci.device_id != 0x2415) + goto device_not_found; + + AC97.nam = AC97.pci.bar[0] & 0xFFFFFF00; + AC97.nabm = AC97.pci.bar[1] & 0xFFFFFF00; + + // Enable port IO, disable MMIO + PCIWriteU8(j.u8[2], j.u8[1], j.u8[0], 0x4, 5); + + OutU32(AC97.nabm + GLOBAL_CTL, 0x03); + OutU16(AC97.nam + RESET, 0xFFFF); + + // Set PCM Output to Max volume + OutU16(AC97.nam + PCM_VOL, 0x0000); + + // Allocate Buffer Descriptor Lists + AC97.bdl[PCM_IN] = CAllocAligned(sizeof(@ac97_bdl), 4096, Fs->code_heap); + AC97.bdl[PCM_OUT] = CAllocAligned(sizeof(@ac97_bdl), 4096, Fs->code_heap); + AC97.bdl[MIC_IN] = CAllocAligned(sizeof(@ac97_bdl), 4096, Fs->code_heap); + + for (i = 0; i < MAX_BDLS; i++) { + AC97.bdl[PCM_OUT]->entries[i].addr = CAllocAligned(PCM_BUF_SIZE, 4096, Fs->code_heap); + AC97.bdl[PCM_OUT]->entries[i].length = BDL_BUF_SIZE / 2; + AC97.bdl[PCM_OUT]->entries[i].flags = 1 << 15; + } + + // Set addresses of Buffer Descriptor Lists + // OutU32(AC97.nabm + PCM_INPUT_REG_BOX + BUFFER_DSC_ADDR, AC97.bdl[PCM_IN]); + OutU32(AC97.nabm + PCM_OUTPUT_REG_BOX + BUFFER_DSC_ADDR, AC97.bdl[PCM_OUT]); + // OutU32(AC97.nabm + MIC_INPUT_REG_BOX + BUFFER_DSC_ADDR, AC97.bdl[MIC_IN]); + + // Set Master Volume + OutU16(AC97.nam + MASTER_VOL, 0x0F0F); + + // Stop playing sound + OutU8(AC97.nabm + PCM_OUTPUT_REG_BOX + BUFFER_CNT, 0); + + // Fill one buffers + @ac97_fill_buffer; + + // Enable interrupt handler + @pci_register_int_handler(&@ac97_int_handler); + + // Start playing sound + OutU8(AC97.nabm + PCM_OUTPUT_REG_BOX + BUFFER_CNT, 1); + + return 0; +} + +@ac97_init; + +"ac97 "; \ No newline at end of file diff --git a/System/Drivers/Audio.HC b/System/Drivers/Audio.HC new file mode 100644 index 0000000..532cfde --- /dev/null +++ b/System/Drivers/Audio.HC @@ -0,0 +1,117 @@ +extern U0 (*fp_snd_fill_buf)(U32* buf, I64 buf_num); + +#define AUDIO_MAX_STREAMS 16 + +#define AUDIO_OUTPUT_BUFFER_SIZE 1024 + +#define AUDIO_STREAM_FIFO_SIZE 1048576 +#define AUDIO_STREAM_TYPE_INPUT 0 +#define AUDIO_STREAM_TYPE_OUTPUT 1 + +class Sound { + // For simplicity, all samples will be converted to 44100 Hz, 2 channels, 16 + // bit when they are loaded. + I64 rate; + I64 channels; + I64 bits; + U32* data; + I64 length; // in samples +}; + +class @audio_device +{ + Bool enabled; +}; + +class @audio_mixer +{ + I64 left; + I64 right; +}; + +class @audio_stream +{ + I64 type; + I64 rate; + I64 channels; + I64 bits; + CFifoI64* data; +}; + +class @audio_wave_generator +{ + F64 duration; + I64 frequency; +}; + +class @audio +{ + I64 driver; + @audio_device device; + @audio_mixer mixer; + @audio_stream output[AUDIO_MAX_STREAMS + 1]; + @audio_wave_generator wavegen; + U0 (*Beep)(); + U0 (*Init)(); + U0 (*MixOutput)(U64 buf, I64); + Sound (*SoundFromFile)(U8* filename); + U0 (*FreeSound)(Sound* snd); + I64 (*PlaySound)(Sound* snd); +}; + +@audio Audio; + +U0 @audio_mix_output(U32* buf, I64 length = NULL) +{ + I64 i; + I64 j; + I64 acc_sample_L = 0; + I64 acc_sample_R = 0; + I64 acc_streams = 0; + U32 sample; + if (!length) + length = AUDIO_OUTPUT_BUFFER_SIZE; + for (i = 0; i < length / 4; i++) { + acc_sample_L = 0; + acc_sample_R = 0; + acc_streams = 0; + if (Audio.wavegen.frequency) { + sample.i16[0] = T(Sin(Audio.wavegen.frequency * Audio.wavegen.duration) >= 0.0, + I16_MAX / 8, I16_MIN / 8); + sample.i16[1] = sample.i16[0]; + FifoI64Ins(Audio.output[AUDIO_MAX_STREAMS].data, sample); + Audio.wavegen.duration += 6.4 / 48000.0; + } + for (j = 0; j < AUDIO_MAX_STREAMS + 1; j++) { + if (FifoI64Cnt(Audio.output[j].data)) { + FifoI64Rem(Audio.output[j].data, &sample); + acc_streams++; + acc_sample_L += sample.i16[0]; + acc_sample_R += sample.i16[1]; + } + } + buf[i].i16[0] = ToI64(acc_sample_L / Sqrt(acc_streams) * Audio.mixer.left / 100); + buf[i].i16[1] = ToI64(acc_sample_R / Sqrt(acc_streams) * Audio.mixer.right / 100); + } +} + +U0 @audio_init() +{ + I64 i = 0; + for (i = 0; i < AUDIO_MAX_STREAMS + 1; i++) + Audio.output[i].data = FifoI64New(AUDIO_STREAM_FIFO_SIZE); + Audio.mixer.left = 100; + Audio.mixer.right = 100; + Audio.wavegen.duration = 0.0; + Audio.wavegen.frequency = 0; + Audio.device.enabled = TRUE; +} + +Audio.driver = NULL; +Audio.Init = &@audio_init; +Audio.MixOutput = &@audio_mix_output; + +// Initialize Audio +Audio.Init(); + +"audio "; \ No newline at end of file diff --git a/System/Drivers/Display.HC b/System/Drivers/Display.HC new file mode 100644 index 0000000..cc97257 --- /dev/null +++ b/System/Drivers/Display.HC @@ -0,0 +1,26 @@ +class @display +{ + I64 width; + I64 height; + I64 bpp; + I64 driver; + U64 fb; + + U0(*Init) + (I64 width, I64 height, I64 bpp, I64 driver); + I64(*Width) + (); + I64(*Height) + (); + I64(*Bpp) + (); + I64(*Driver) + (); + U0 (*Update)(); +}; + +@display Display; +Display.driver = NULL; +Display.Update = NULL; + +"display "; diff --git a/System/Drivers/Mouse.HC b/System/Drivers/Mouse.HC new file mode 100644 index 0000000..763b61d --- /dev/null +++ b/System/Drivers/Mouse.HC @@ -0,0 +1,111 @@ +// FIXME: This should be an Input driver which contains both Keyboard/Mouse +// classes. +extern U0 @vmsvga_mouse_pointer_set(U32* pointer, I64 width, I64 height); + +#define MI_QEMU 0x01 +#define MI_VBOX 0x02 +#define MS_UPDATE_INTERVAL 10 + +#define MS_UP 0 +#define MS_DOWN 1 + +class @mouse +{ + I64 x; + I64 y; + I64 z; + I64 delta_z; + I64 wheel_sensitivity; + U32* pointer; + I64 integration_type; + Bool left; + Bool right; + Bool natural_scroll; + I64 (*X)(); + I64 (*Y)(); + U0 (*PointerSet)(U32* pointer, I64 width, I64 height); + U0 (*Init)(); + U0 (*Update)(); + U0 (*Task)(); +}; + +class @keyboard +{ + I64 active_key; + I64 active_key_tS; + I64 last_key_tS; + U0 (*Update)(); +}; + +@mouse Mouse; +@keyboard Keyboard; + +U0 @keyboard_update() +{ + I64 sc; + if (FifoI64Rem(kbd.scan_code_fifo, &sc)) { + if (!(sc & SCF_KEY_UP)) { + Keyboard.active_key = sc(U8); + Keyboard.active_key_tS = cnts.jiffies; + return; + } + } + Keyboard.active_key = 0; +} + +Keyboard.Update = &@keyboard_update; + +I64 @mouse_x() { return Mouse.x; } + +I64 @mouse_y() { return Mouse.y; } + +U0 @mouse_integration_type_set(I64 type) { Mouse.integration_type = type; } + +U0 @mouse_pointer_set(U32* pointer, I64 width, I64 height) +{ + if (Mouse.pointer != pointer) { + Mouse.pointer = pointer; + if (Mouse.integration_type == MI_VBOX) { + @vmsvga_mouse_pointer_set(pointer, width, height); + } + } +} + +U0 @mouse_init() +{ + Mouse.x = Display.Width() / 2; + Mouse.y = Display.Height() / 2; + Mouse.z = ms.pos.z; + Mouse.wheel_sensitivity = 2; + Mouse.pointer = NULL; + Mouse.left = OFF; + Mouse.right = OFF; +} + +U0 @mouse_task() +{ + while (1) { + WinMsUpdate; + KbdMsHndlr(0, 0); + Keyboard.Update(); + if (Mouse.Update) + Mouse.Update(); + if (!Mouse.Update) { + // Mouse.x = ms.pos.x; + // Mouse.y = ms.pos.y; + Mouse.left = ms.lb > 0; + Mouse.right = ms.rb > 0; + } + Mouse.z = ms.pos.z; + Sleep(MS_UPDATE_INTERVAL); + } +} + +Mouse.X = &@mouse_x; +Mouse.Y = &@mouse_y; +Mouse.PointerSet = &@mouse_pointer_set; +Mouse.Init = &@mouse_init; +Mouse.Update = NULL; +Mouse.Task = &@mouse_task; + +"mouse "; diff --git a/System/Drivers/Pci.HC b/System/Drivers/Pci.HC new file mode 100644 index 0000000..3d84393 --- /dev/null +++ b/System/Drivers/Pci.HC @@ -0,0 +1,98 @@ +#define PCI_INTH_MAX 16 + +U64 @pci_int_handler[PCI_INTH_MAX]; + +class @pci_info +{ + U16 vendor_id; + U16 device_id; + U16 command; + U16 status; + U32 _class; + U32 bar[6]; + U32 cap_pointer; +}; + +class @pci_cap +{ + U8 cap_vndr; /* Generic PCI field: PCI_CAP_ID_VNDR */ + U8 cap_next; /* Generic PCI field: next ptr. */ + U8 cap_len; /* Generic PCI field: capability length */ + U8 cfg_type; /* Identifies the structure. */ + U8 bar; /* Where to find it. */ + U8 padding[3]; /* Pad to full dword. */ + U32 offset; /* Offset within bar. */ + U32 length; /* Length of the structure, in bytes. */ +}; + +U0 @get_pci_info(I64 i, @pci_info* pci) +{ + I64 j; + pci->vendor_id = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], 0x0) & 0xFFFF; + pci->device_id = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], 0x0) >> 16; + pci->command = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], 0x4) & 0xFFFF; + pci->status = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], 0x4) >> 16; + pci->_class = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], 0x8) >> 24; + for (j = 0; j < 6; j++) + pci->bar[j] = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], 0x10 + (0x04 * j)); +} + +U0 @get_pci_cap(I64 i, @pci_cap* cap, I64 idx) +{ + I64 base = 0x40 + (idx * 16); + U32 u32; + u32 = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], base); + cap->cap_vndr = u32.u8[0]; + cap->cap_next = u32.u8[1]; + cap->cap_len = u32.u8[2]; + cap->cfg_type = u32.u8[3]; + u32 = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], base + 0x04); + cap->bar = u32.u8[0]; + cap->offset = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], base + 0x08); + cap->length = PCIReadU32(i.u8[2], i.u8[1], i.u8[0], base + 0x0c); +} + +U0 @pci_reroute_interrupts(I64 base, I64 cpu) +{ + I64 i; + U8* da = dev.uncached_alias + IOAPIC_REG; + U32* _d = dev.uncached_alias + IOAPIC_DATA; + + for (i = 0; i < 4; i++) { + *da = IOREDTAB + i * 2 + 1; + *_d = dev.mp_apic_ids[cpu] << 24; + *da = IOREDTAB + i * 2; + *_d = 0x4000 + base + i; + } +} + +I64 @pci_register_int_handler(U64 handler) +{ + if (!handler) + return -1; + I64 i = 0; + while (@pci_int_handler[i]) + i++; + if (i > PCI_INTH_MAX - 1) + return -1; + @pci_int_handler[i] = handler; + return 0; +} + +interrupt U0 @pci_interrupt_handler() +{ + I64 i; + for (i = 0; i < PCI_INTH_MAX; i++) + if (@pci_int_handler[i]) + Call(@pci_int_handler[i]); + *(dev.uncached_alias + LAPIC_EOI)(U32*) = 0; +} + +MemSet(&@pci_int_handler, NULL, sizeof(U64) * PCI_INTH_MAX); +IntEntrySet(0x40, &@pci_interrupt_handler, IDTET_IRQ); +IntEntrySet(0x41, &@pci_interrupt_handler, IDTET_IRQ); +IntEntrySet(0x42, &@pci_interrupt_handler, IDTET_IRQ); +IntEntrySet(0x43, &@pci_interrupt_handler, IDTET_IRQ); +@pci_reroute_interrupts(0x40, 0); + +"pci "; \ No newline at end of file diff --git a/System/Drivers/VMSVGA.HC b/System/Drivers/VMSVGA.HC new file mode 100644 index 0000000..9e42374 --- /dev/null +++ b/System/Drivers/VMSVGA.HC @@ -0,0 +1,255 @@ +class @vmsvga_info +{ + U16 io_base; + U32* fifo; + U16 width; + U16 height; + U16 bpp; + U64 fb; + U32 capabilities; +}; + +#define VMWGFX_FIFO_STATIC_SIZE (1024 * 1024) + +#define VMSVGA_MAGIC 0x900000 +#define VMSVGA_ID_2 (VMSVGA_MAGIC << 8 | 2) +#define VMSVGA_MOUSE_ID 1 + +#define VMSVGA_CAP_GMR 0x00100000 + +#define VMSVGA_CMD_INVALID_CMD 0 +#define VMSVGA_CMD_UPDATE 1 +#define VMSVGA_CMD_RECT_COPY 3 +#define VMSVGA_CMD_DEFINE_CURSOR 19 +#define VMSVGA_CMD_DEFINE_ALPHA_CURSOR 22 +#define VMSVGA_CMD_UPDATE_VERBOSE 25 +#define VMSVGA_CMD_FRONT_ROP_FILL 29 +#define VMSVGA_CMD_FENCE 30 +#define VMSVGA_CMD_ESCAPE 33 +#define VMSVGA_CMD_DEFINE_SCREEN 34 +#define VMSVGA_CMD_DESTROY_SCREEN 35 +#define VMSVGA_CMD_DEFINE_GMRFB 36 +#define VMSVGA_CMD_BLIT_GMRFB_TO_SCREEN 37 +#define VMSVGA_CMD_BLIT_SCREEN_TO_GMRFB 38 +#define VMSVGA_CMD_ANNOTATION_FILL 39 +#define VMSVGA_CMD_ANNOTATION_COPY 40 +#define VMSVGA_CMD_DEFINE_GMR2 41 +#define VMSVGA_CMD_REMAP_GMR2 42 + +#define VMSVGA_REG_ID 0 +#define VMSVGA_REG_ENABLE 1 +#define VMSVGA_REG_WIDTH 2 +#define VMSVGA_REG_HEIGHT 3 +#define VMSVGA_REG_MAX_WIDTH 4 +#define VMSVGA_REG_MAX_HEIGHT 5 +#define VMSVGA_REG_DEPTH 6 +#define VMSVGA_REG_BITS_PER_PIXEL 7 /* Current bpp in the guest */ +#define VMSVGA_REG_PSEUDOCOLOR 8 +#define VMSVGA_REG_RED_MASK 9 +#define VMSVGA_REG_GREEN_MASK 10 +#define VMSVGA_REG_BLUE_MASK 11 +#define VMSVGA_REG_BYTES_PER_LINE 12 +#define VMSVGA_REG_FB_START 13 /* (Deprecated) */ +#define VMSVGA_REG_FB_OFFSET 14 +#define VMSVGA_REG_VRAM_SIZE 15 +#define VMSVGA_REG_FB_SIZE 16 + +/* ID 0 implementation only had the above registers then the palette */ +#define VMSVGA_REG_CAPABILITIES 17 +#define VMSVGA_REG_MEM_START 18 /* (Deprecated) */ +#define VMSVGA_REG_MEM_SIZE 19 +#define VMSVGA_REG_CONFIG_DONE 20 /* Set when memory area configured */ +#define VMSVGA_REG_SYNC 21 /* See "FIFO Synchronization Registers" */ +#define VMSVGA_REG_BUSY 22 /* See "FIFO Synchronization Registers" */ +#define VMSVGA_REG_GUEST_ID 23 /* Set guest OS identifier */ +#define VMSVGA_REG_CURSOR_ID 24 /* (Deprecated) */ +#define VMSVGA_REG_CURSOR_X 25 /* (Deprecated) */ +#define VMSVGA_REG_CURSOR_Y 26 /* (Deprecated) */ +#define VMSVGA_REG_CURSOR_ON 27 /* (Deprecated) */ +#define VMSVGA_REG_HOST_BITS_PER_PIXEL 28 /* (Deprecated) */ +#define VMSVGA_REG_SCRATCH_SIZE 29 /* Number of scratch registers */ +#define VMSVGA_REG_MEM_REGS 30 /* Number of FIFO registers */ +#define VMSVGA_REG_NUM_DISPLAYS 31 /* (Deprecated) */ +#define VMSVGA_REG_PITCHLOCK 32 /* Fixed pitch for all modes */ +#define VMSVGA_REG_IRQMASK 33 /* Interrupt mask */ + +/* Legacy multi-monitor support */ +#define VMSVGA_REG_NUM_GUEST_DISPLAYS \ + 34 /* Number of guest displays in X/Y direction */ +#define VMSVGA_REG_DISPLAY_ID \ + 35 /* Display ID for the following display attributes */ +#define VMSVGA_REG_DISPLAY_IS_PRIMARY \ + 36 /* Whether this is a primary display \ + */ +#define VMSVGA_REG_DISPLAY_POSITION_X 37 /* The display position x */ +#define VMSVGA_REG_DISPLAY_POSITION_Y 38 /* The display position y */ +#define VMSVGA_REG_DISPLAY_WIDTH 39 /* The display's width */ +#define VMSVGA_REG_DISPLAY_HEIGHT 40 /* The display's height */ + +/* See "Guest memory regions" below. */ +#define VMSVGA_REG_GMR_ID 41 +#define VMSVGA_REG_GMR_DESCRIPTOR 42 +#define VMSVGA_REG_GMR_MAX_IDS 43 +#define VMSVGA_REG_GMR_MAX_DESCRIPTOR_LENGTH 44 +#define VMSVGA_REG_TRACES \ + 45 /* Enable trace-based updates even when FIFO is on */ +#define VMSVGA_REG_GMRS_MAX_PAGES \ + 46 /* Maximum number of 4KB pages for all GMRs */ +#define VMSVGA_REG_MEMORY_SIZE \ + 47 /* Total dedicated device memory excluding FIFO */ +#define VMSVGA_REG_TOP 48 /* Must be 1 more than the last register */ + +#define VMSVGA_FIFO_MIN 0 +#define VMSVGA_FIFO_MAX 1 +#define VMSVGA_FIFO_NEXT_CMD 2 +#define VMSVGA_FIFO_STOP 3 +#define VMSVGA_FIFO_CAPABILITIES 4 +#define VMSVGA_FIFO_FLAGS 5 +#define VMSVGA_FIFO_FENCE 6 +#define VMSVGA_FIFO_3D_HWVERSION 7 +#define VMSVGA_FIFO_PITCHLOCK 8 +#define VMSVGA_FIFO_CURSOR_ON 9 +#define VMSVGA_FIFO_CURSOR_X 10 +#define VMSVGA_FIFO_CURSOR_Y 11 +#define VMSVGA_FIFO_CURSOR_COUNT 12 +#define VMSVGA_FIFO_CURSOR_LAST_UPDATED 13 +#define VMSVGA_FIFO_RESERVED 14 +#define VMSVGA_FIFO_CURSOR_SCREEN_ID 15 +#define VMSVGA_FIFO_DEAD 16 +#define VMSVGA_FIFO_3D_HWVERSION_REVISED 17 +#define VMSVGA_FIFO_3D_CAPS 18 +#define VMSVGA_FIFO_3D_CAPS_LAST = 19 +#define VMSVGA_FIFO_GUEST_3D_HWVERSION 20 +#define VMSVGA_FIFO_FENCE_GOAL 21 +#define VMSVGA_FIFO_BUSY 22 +#define VMSVGA_FIFO_NUM_REGS 23 + +#define VMSVGA_FIFO_CAP_NONE 0 +#define VMSVGA_FIFO_CAP_FENCE (1 << 0) +#define VMSVGA_FIFO_CAP_ACCELFRONT (1 << 1) +#define VMSVGA_FIFO_CAP_PITCHLOCK (1 << 2) +#define VMSVGA_FIFO_CAP_VIDEO (1 << 3) +#define VMSVGA_FIFO_CAP_CURSOR_BYPASS_3 (1 << 4) +#define VMSVGA_FIFO_CAP_ESCAPE (1 << 5) +#define VMSVGA_FIFO_CAP_RESERVE (1 << 6) +#define VMSVGA_FIFO_CAP_SCREEN_OBJECT (1 << 7) +#define VMSVGA_FIFO_CAP_GMR2 (1 << 8) +// #define VMSVGA_FIFO_CAP_3D_HWVERSION_REVISED VMSVGA_FIFO_CAP_GMR2 +#define VMSVGA_FIFO_CAP_SCREEN_OBJECT_2 (1 << 9) +#define VMSVGA_FIFO_CAP_DEAD (1 << 10) + +@vmsvga_info vmsvga; +MemSet(&vmsvga, 0, sizeof(@vmsvga_info)); + +U32 @vmsvga_reg_read(I64 index) +{ + OutU32(vmsvga.io_base, index); + return InU32(vmsvga.io_base + 1); +} + +U0 @vmsvga_reg_write(I64 index, U32 val) +{ + OutU32(vmsvga.io_base, index); + OutU32(vmsvga.io_base + 1, val); +} + +U0 @vmsvga_fifo_write(U32 value) +{ + /* Need to sync? */ + if ((vmsvga.fifo[VMSVGA_FIFO_NEXT_CMD] + sizeof(U32) == vmsvga.fifo[VMSVGA_FIFO_STOP]) || (vmsvga.fifo[VMSVGA_FIFO_NEXT_CMD] == vmsvga.fifo[VMSVGA_FIFO_MAX] - sizeof(U32) && vmsvga.fifo[VMSVGA_FIFO_STOP] == vmsvga.fifo[VMSVGA_FIFO_MIN])) { + //"Syncing because of full fifo\n"; + // vmwareWaitForFB(pVMWARE); + } + vmsvga.fifo[vmsvga.fifo[VMSVGA_FIFO_NEXT_CMD] / sizeof(U32)] = value; + if (vmsvga.fifo[VMSVGA_FIFO_NEXT_CMD] == vmsvga.fifo[VMSVGA_FIFO_MAX] - sizeof(U32)) { + vmsvga.fifo[VMSVGA_FIFO_NEXT_CMD] = vmsvga.fifo[VMSVGA_FIFO_MIN]; + } else { + vmsvga.fifo[VMSVGA_FIFO_NEXT_CMD] += sizeof(U32); + } +} + +U0 @vmsvga_fifo_get_cap(U8* s, I64 cap) +{ + "%32s:", s; + if ((vmsvga.fifo[VMSVGA_FIFO_CAPABILITIES] & cap) == cap) + "%s\n", "True"; + else + "%s\n", "False"; +} + +I64 @vmsvga_init(I64 w, I64 h, I64 bpp) +{ + I64 j; + j = PCIClassFind(0x030000, 0); + if (j < 0) { + //"VMSVGA device not found.\n"; + return -1; + } + vmsvga.io_base = PCIReadU16(j.u8[2], j.u8[1], j.u8[0], 0x10) & ~(0x0F); + @vmsvga_reg_write(VMSVGA_REG_ID, VMSVGA_ID_2); + if (@vmsvga_reg_read(VMSVGA_REG_ID) == VMSVGA_ID_2) { + //"VMSVGA driver version 2 supported.\n"; + } else { + //"VMSVGA device not supported.\n"; + return -1; + } + vmsvga.width = w; + vmsvga.height = h; + vmsvga.bpp = bpp; + vmsvga.fb = @vmsvga_reg_read(VMSVGA_REG_FB_START); + vmsvga.fifo = @vmsvga_reg_read(VMSVGA_REG_MEM_START); + //"FIFO @ 0x%08X (%d bytes)\n", vmsvga.fifo, + // @vmsvga_reg_read(VMSVGA_REG_MEM_SIZE); + @vmsvga_reg_write(VMSVGA_REG_WIDTH, 1920); + @vmsvga_reg_write(VMSVGA_REG_HEIGHT, 1080); + @vmsvga_reg_write(VMSVGA_REG_BITS_PER_PIXEL, 32); + @vmsvga_reg_write(VMSVGA_REG_ENABLE, 1); + vmsvga.fifo[VMSVGA_FIFO_MIN] = 16; + vmsvga.fifo[VMSVGA_FIFO_MAX] = 16 + (10 * 1024); + vmsvga.fifo[VMSVGA_FIFO_NEXT_CMD] = 16; + vmsvga.fifo[VMSVGA_FIFO_STOP] = 16; + @vmsvga_reg_write(VMSVGA_REG_CONFIG_DONE, 0); + @vmsvga_fifo_write(VMSVGA_CMD_UPDATE); + @vmsvga_fifo_write(0); + @vmsvga_fifo_write(0); + @vmsvga_fifo_write(0); + @vmsvga_fifo_write(0); + @vmsvga_reg_write(VMSVGA_REG_CONFIG_DONE, 1); + return 0; +} + +U0 @vmsvga_mouse_pointer_set(U32* pointer, I64 width, I64 height) +{ + @vmsvga_reg_write(VMSVGA_REG_CONFIG_DONE, 0); + @vmsvga_fifo_write(VMSVGA_CMD_DEFINE_ALPHA_CURSOR); + @vmsvga_fifo_write(VMSVGA_MOUSE_ID); + @vmsvga_fifo_write(0); + @vmsvga_fifo_write(0); + @vmsvga_fifo_write(width); + @vmsvga_fifo_write(height); + I64 x, y; + for (y = 0; y < height; y++) { + for (x = 0; x < width; x++) { + @vmsvga_fifo_write(pointer[(y * width) + x]); + } + } + @vmsvga_reg_write(VMSVGA_REG_CONFIG_DONE, 1); +} + +U64 @vmsvga_get_framebuffer() { return vmsvga.fb; } + +U0 @vmsvga_display_update() { @vmsvga_reg_write(VMSVGA_REG_ENABLE, 1); } + +class @vmsvga +{ + U0(*Init) + (I64 w, I64 h, I64 bpp); + U64(*FrameBuffer) + (); +}; + +@vmsvga VMSVGA; +VMSVGA.FrameBuffer = &@vmsvga_get_framebuffer; +VMSVGA.Init = &@vmsvga_init; + +"vmsvga "; diff --git a/System/Drivers/VMwareTools.HC b/System/Drivers/VMwareTools.HC new file mode 100644 index 0000000..34125f1 --- /dev/null +++ b/System/Drivers/VMwareTools.HC @@ -0,0 +1,148 @@ +// https://wiki.osdev.org/VMware_tools + +#define VMWARE_MAGIC 0x564D5868 +#define VMWARE_PORT 0x5658 +#define VMWARE_PORTHB 0x5659 + +#define CMD_GETVERSION 10 + +#define CMD_ABSPOINTER_DATA 39 +#define CMD_ABSPOINTER_STATUS 40 +#define CMD_ABSPOINTER_COMMAND 41 + +#define ABSPOINTER_ENABLE 0x45414552 +#define ABSPOINTER_RELATIVE 0xF5 +#define ABSPOINTER_ABSOLUTE 0x53424152 + +class @vmware_cmd +{ + union { + U32 ax; + U32 magic; + }; + union { + U32 bx; + I32 size; + }; + union { + U32 cx; + U16 command; + }; + union { + U32 dx; + U16 port; + }; + U32 si; + U32 di; +}; + +U0 @vmware_send(@vmware_cmd* cmd) +{ + U32 reg RAX reg_ax = cmd->ax; + U32 reg RBX reg_bx = cmd->bx; + U32 reg RCX reg_cx = cmd->cx; + U32 reg RDX reg_dx = cmd->dx; + U32 reg RSI reg_si = cmd->si; + U32 reg RDI reg_di = cmd->di; + reg_ax = VMWARE_MAGIC; + reg_dx = VMWARE_PORT; + asm { + IN AX, DX + } + cmd->ax = reg_ax; + cmd->bx = reg_bx; + cmd->cx = reg_cx; + cmd->dx = reg_dx; + cmd->si = reg_si; + cmd->di = reg_di; +} + +Bool @vmware_backdoor_is_present() +{ + @vmware_cmd cmd; + cmd.bx = ~VMWARE_MAGIC; + cmd.command = CMD_GETVERSION; + @vmware_send(&cmd); + if (cmd.bx != VMWARE_MAGIC || cmd.ax == 0xFFFFFFFF) { + return FALSE; + } + return TRUE; +} + +U0 @vmware_ms_absolute() +{ + @vmware_cmd cmd; + + cmd.bx = ABSPOINTER_ENABLE; + cmd.command = CMD_ABSPOINTER_COMMAND; + @vmware_send(&cmd); + + cmd.bx = 0; + cmd.command = CMD_ABSPOINTER_STATUS; + @vmware_send(&cmd); + + cmd.bx = 1; + cmd.command = CMD_ABSPOINTER_DATA; + @vmware_send(&cmd); + + cmd.bx = ABSPOINTER_ABSOLUTE; + cmd.command = CMD_ABSPOINTER_COMMAND; + @vmware_send(&cmd); +} + +U0 @vmware_handle_mouse() +{ + @vmware_cmd cmd; + cmd.bx = 0; + cmd.command = CMD_ABSPOINTER_STATUS; + @vmware_send(&cmd); + + if (cmd.ax == 0xFFFF0000) { + @vmware_ms_absolute; + return; + } + + if ((cmd.ax & 0xFFFF) < 4) + return; + + cmd.bx = 4; + cmd.command = CMD_ABSPOINTER_DATA; + @vmware_send(&cmd); + + I32 buttons = (cmd.ax & 0xFFFF); + I64 z = cmd.dx; + if (z > 1) { + z = -1; + } + + MsSet(@lerp(cmd.bx, 0xffff, Display.width), @lerp(cmd.cx, 0xffff, Display.height), ms.pos.z + z); + Mouse.x = @lerp(cmd.bx, 0xffff, Display.width); + Mouse.y = @lerp(cmd.cx, 0xffff, Display.height); + // MsSet((cmd.bx * Display.width) / 0xffff, (cmd.cx * Display.height) / 0xffff, ms.pos.z + z); + ms.lb = buttons & 0x20; + ms.rb = buttons & 0x10; +} + +U0 @vmware_ms_nop() { } + +U0 @vmware_ms_update() +{ + while (1) { + @vmware_handle_mouse; + Sleep(1); + } +} + +U0 @vmware_tools_init() +{ + if (!@vmware_backdoor_is_present) { + return; + } + @patch_jmp_rel32(&WinMsUpdate, &@vmware_ms_nop); + @vmware_ms_absolute; + Spawn(&@vmware_ms_update, , "VMwareMsUpdateTask"); +} + +@vmware_tools_init; + +"vmware-tools "; diff --git a/System/Drivers/Virtio-blk.HC b/System/Drivers/Virtio-blk.HC new file mode 100644 index 0000000..d5762aa --- /dev/null +++ b/System/Drivers/Virtio-blk.HC @@ -0,0 +1,475 @@ +// Virtio.HC + +// +// PCI virtio I/O registers. +// + +#define VIRTIO_PCI_HOST_FEATURES 0 // Features supported by the host +#define VIRTIO_PCI_GUEST_FEATURES 4 // Features activated by the guest +#define VIRTIO_PCI_QUEUE_PFN 8 // PFN for the currently selected queue +#define VIRTIO_PCI_QUEUE_SIZE 12 // Queue size for the currently selected queue +#define VIRTIO_PCI_QUEUE_SEL 14 // Queue selector +#define VIRTIO_PCI_QUEUE_NOTIFY 16 // Queue notifier +#define VIRTIO_PCI_STATUS 18 // Device status register +#define VIRTIO_PCI_ISR 19 // Interrupt status register +#define VIRTIO_PCI_CONFIG 20 // Configuration data block + +// +// PCI virtio status register bits +// + +#define VIRTIO_CONFIG_S_ACKNOWLEDGE 1 +#define VIRTIO_CONFIG_S_DRIVER 2 +#define VIRTIO_CONFIG_S_DRIVER_OK 4 +#define VIRTIO_CONFIG_S_FAILED 0x80 + +// +// Ring descriptor flags +// + +#define VRING_DESC_F_NEXT 1 // Buffer continues via the next field +#define VRING_DESC_F_WRITE 2 // Buffer is write-only (otherwise read-only) +#define VRING_DESC_F_INDIRECT 4 // Buffer contains a list of buffer descriptors + +class @virtio_queue_buf +{ + U64 address; + U32 length; + U16 flags; + U16 next; +}; +class @virtio_avail +{ + U16 flags; + U16 index; + U16 ring[256]; + U16 int_index; +}; +class @virtio_used_item +{ + U32 index; + U32 length; +}; +class @virtio_used +{ + U16 flags; + U16 index; + @virtio_used_item ring[256]; + U16 int_index; +}; +class @virtio_queue +{ + @virtio_queue_buf buffers[256]; + @virtio_avail available; + U8 padding[3578]; + @virtio_used used; +}; + +class @virtio_avail_buf +{ + U32 index; + U64 address; + U32 length; +}; + +class @virtio_buf_info +{ + U8* buffer; + U64 size; + U8 flags; + + // If the user wants to keep same buffer as passed in this struct, use "true". + // otherwise, the supplied buffer will be copied in the queues' buffer + Bool copy; +}; + +// Virtio-blk.HC + +#define BDT_VIRTIO_BLK 10 + +#define VIRTIO_BLK_T_IN 0 +#define VIRTIO_BLK_T_OUT 1 +#define VIRTIO_BLK_T_FLUSH 4 + +#define VIRTIO_BLK_MAX_BLK 0x400000 // Limit blkdev to 2G max, set to NULL to use entire disk (not recommended for RedSea) + +class @virtio_blk +{ + U16 port; + U32 blks; + @virtio_queue* vq; + I64 vq_size; + I64 vq_index; + U8 status; +}; + +class @virtio_blk_request +{ + U32 type; + U32 priority; + U64 sector; +}; + +@virtio_blk virtio_blk; +MemSet(&virtio_blk, 0, sizeof(@virtio_blk)); + +I64 VirtioBlkInit() +{ + I64 j; + + // Scan for device + j = PCIClassFind(0x010000, 0); + if (j < 0) { + "\n[virtio-blk] No device found\n"; + return -1; + } + virtio_blk.port = PCIReadU32(j.u8[2], + j.u8[1], j.u8[0], 0x10) + & 0xFFFFFFFC; + + virtio_blk.blks = InU32(virtio_blk.port + VIRTIO_PCI_CONFIG); + + // Reset Device + OutU8(virtio_blk.port + VIRTIO_PCI_STATUS, 0); + + // Found Driver + OutU8(virtio_blk.port + VIRTIO_PCI_STATUS, InU8(virtio_blk.port + VIRTIO_PCI_STATUS) | VIRTIO_CONFIG_S_ACKNOWLEDGE | VIRTIO_CONFIG_S_DRIVER); + + // Set up virt queue + OutU16(virtio_blk.port + VIRTIO_PCI_QUEUE_SEL, 0); + virtio_blk.vq_size = InU16(virtio_blk.port + VIRTIO_PCI_QUEUE_SIZE); // 256 + virtio_blk.vq = CAllocAligned(sizeof(@virtio_queue), 4096, erythros_mem_task->code_heap); + OutU32(virtio_blk.port + VIRTIO_PCI_QUEUE_PFN, virtio_blk.vq / 4096); + + // Init OK + OutU8(virtio_blk.port + VIRTIO_PCI_STATUS, InU8(virtio_blk.port + VIRTIO_PCI_STATUS) | VIRTIO_CONFIG_S_DRIVER_OK); + virtio_blk.vq_index = 0; +} + +// DskVIO.HC + +U0 VIOFlush() +{ + I64 j; + I64 vq_idx; + @virtio_blk_request* brq = CAlloc(sizeof(@virtio_blk_request), erythros_mem_task); + brq->type = VIRTIO_BLK_T_FLUSH; + brq->sector = NULL; + vq_idx = virtio_blk.vq->available.index % 256; + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].address = brq; + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].length = sizeof(@virtio_blk_request); + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].flags = VRING_DESC_F_NEXT; + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].next = (virtio_blk.vq_index + 1) % 256; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].address = &virtio_blk.status; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].length = 1; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].flags = VRING_DESC_F_WRITE; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].next = 0; + virtio_blk.vq->available.ring[vq_idx] = virtio_blk.vq_index % 256; + virtio_blk.vq_index += 2; + j = virtio_blk.vq->used.index; + virtio_blk.vq->available.index++; + OutU16(virtio_blk.port + VIRTIO_PCI_QUEUE_NOTIFY, 0); + while (j == virtio_blk.vq->used.index) { + Yield; + } + Free(brq); +} + +Bool VIORBlks(CDrv* dv, U8* buf, I64 blk, I64 cnt) +{ + no_warn dv; + I64 i, j; + I64 vq_idx; + U64 addr; + @virtio_blk_request* brq = CAlloc(sizeof(@virtio_blk_request), erythros_mem_task); + for (i = 0; i < cnt; i++) { + brq->type = VIRTIO_BLK_T_IN; + brq->sector = blk + i; + vq_idx = virtio_blk.vq->available.index % 256; + addr = buf + (BLK_SIZE * i); + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].address = brq; + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].length = sizeof(@virtio_blk_request); + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].flags = VRING_DESC_F_NEXT; + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].next = (virtio_blk.vq_index + 1) % 256; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].address = addr; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].length = BLK_SIZE; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].flags = VRING_DESC_F_WRITE | VRING_DESC_F_NEXT; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].next = (virtio_blk.vq_index + 2) % 256; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 2) % 256].address = &virtio_blk.status; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 2) % 256].length = 1; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 2) % 256].flags = VRING_DESC_F_WRITE; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 2) % 256].next = 0; + virtio_blk.vq->available.ring[vq_idx] = virtio_blk.vq_index % 256; + virtio_blk.vq_index += 3; + j = virtio_blk.vq->used.index; + virtio_blk.vq->available.index++; + OutU16(virtio_blk.port + VIRTIO_PCI_QUEUE_NOTIFY, 0); + while (j == virtio_blk.vq->used.index) { + Yield; + } + } + Free(brq); + return TRUE; +} + +Bool VIOWBlks(CDrv* dv, U8* buf, I64 blk, I64 cnt) +{ + no_warn dv; + I64 i, j; + I64 vq_idx; + U64 addr; + @virtio_blk_request* brq = CAlloc(sizeof(@virtio_blk_request), erythros_mem_task); + for (i = 0; i < cnt; i++) { + brq->type = VIRTIO_BLK_T_OUT; + brq->sector = blk + i; + vq_idx = virtio_blk.vq->available.index % 256; + addr = buf + (BLK_SIZE * i); + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].address = brq; + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].length = sizeof(@virtio_blk_request); + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].flags = VRING_DESC_F_NEXT; + virtio_blk.vq->buffers[virtio_blk.vq_index % 256].next = (virtio_blk.vq_index + 1) % 256; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].address = addr; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].length = BLK_SIZE; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].flags = VRING_DESC_F_NEXT; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 1) % 256].next = (virtio_blk.vq_index + 2) % 256; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 2) % 256].address = &virtio_blk.status; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 2) % 256].length = 1; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 2) % 256].flags = VRING_DESC_F_WRITE; + virtio_blk.vq->buffers[(virtio_blk.vq_index + 2) % 256].next = 0; + virtio_blk.vq->available.ring[vq_idx] = virtio_blk.vq_index % 256; + virtio_blk.vq_index += 3; + j = virtio_blk.vq->used.index; + virtio_blk.vq->available.index++; + OutU16(virtio_blk.port + VIRTIO_PCI_QUEUE_NOTIFY, 0); + while (j == virtio_blk.vq->used.index) { + Yield; + } + } + Free(brq); + VIOFlush; + return TRUE; +} + +U0 RedSeaTryInit(CDrv* dv) +{ + CRedSeaBoot br; + Bool unlock; + try { + unlock = DrvLock(dv); + BlkRead(dv, &br, dv->drv_offset, 1); + if (br.signature != MBR_PT_REDSEA || br.signature2 != 0xAA55) + return; + dv->fs_type = FSt_REDSEA; + CallExtStr("RedSeaFreeFreeLst", dv); + dv->spc = 1; + dv->size = br.sects; + dv->data_area = dv->drv_offset + br.bitmap_sects; + dv->root_clus = br.root_clus; + dv->fat1 = dv->fat2 = dv->drv_offset + 1; + CallExtStr("DrvFATBlkAlloc", dv); + if (unlock) + DrvUnlock(dv); + } catch if (unlock) + DrvUnlock(dv); +} + +U8 MountVirtioBlk() +{ // Mount Virtio-blk device + CDrv* dv = DrvMakeFreeSlot(DrvNextFreeLet('A')); + CBlkDev* bd = BlkDevNextFreeSlot(dv->drv_let, BDT_RAM); + CRedSeaBoot* bs = CAlloc(BLK_SIZE, erythros_mem_task); + bd->max_blk = 512; + BlkDevAdd(bd, , TRUE, TRUE); + bd->type = BDT_VIRTIO_BLK; + if (VIRTIO_BLK_MAX_BLK) { + bd->max_blk = Min(VIRTIO_BLK_MAX_BLK, virtio_blk.blks); + } else { + bd->max_blk = virtio_blk.blks; + } + Free(bd->RAM_dsk); + dv->size = bd->max_blk + 1 - bd->drv_offset; + VIORBlks(dv, bs, 0, 1); + dv->root_clus = bs->root_clus; + dv->data_area = bs->bitmap_sects; + dv->next_free = NULL; + dv->last_free = NULL; + Free(bs); + RedSeaTryInit(dv); + return dv->drv_let; +} + +// DskBlk2.HC + +Bool BlkRead2(CDrv* dv, U8* buf, I64 blk, I64 cnt) +{ // Read blk cnt from Drv to buf. + Bool res = TRUE, unlock; + CBlkDev* bd = dv->bd; + if (cnt <= 0) + return TRUE; + DrvChk(dv); + try { + unlock = DrvLock(dv); + CallExtStr("BlkDevInit", bd); + if (dv->drv_offset && blk < dv->drv_offset || blk + cnt > dv->drv_offset + dv->size) + throw('Drv'); + if (bd->flags & BDF_READ_CACHE) + CallExtStr("RCache", dv, &buf, &blk, &cnt); + if (cnt > 0) { + switch (bd->type) { + case BDT_RAM: + MemCpy(buf, bd->RAM_dsk + blk << BLK_SIZE_BITS, cnt << BLK_SIZE_BITS); + break; + case BDT_ISO_FILE_READ: + case BDT_ISO_FILE_WRITE: + FBlkRead(bd->file_dsk, buf, blk, cnt); + break; + case BDT_ATA: + case BDT_ATAPI: + res = CallExtStr("ATARBlks", dv, buf, blk, cnt); + break; + case BDT_VIRTIO_BLK: + res = VIORBlks(dv, buf, blk, cnt); + break; + } + bd->last_time = tS; + if (bd->flags & BDF_READ_CACHE) + CallExtStr("DskCacheAdd", dv, buf, blk, cnt); + } + if (unlock) + DrvUnlock(dv); + } catch if (unlock) + DrvUnlock(dv); + return res; +} + +Bool BlkWrite2(CDrv* dv, U8* buf, I64 blk, I64 cnt) +{ // Write blk cnt from buf to Drv. + Bool res = TRUE, unlock; + CBlkDev* bd = dv->bd; + if (cnt <= 0) + return TRUE; + DrvChk(dv); + try { + unlock = DrvLock(dv); + CallExtStr("BlkDevInit", bd); + if (bd->flags & BDF_READ_ONLY && !(bd->flags & BDF_READ_ONLY_OVERRIDE)) + throw('BlkDev'); + if (dv->drv_offset && blk < dv->drv_offset || blk + cnt > dv->drv_offset + dv->size) + throw('Drv'); + if (cnt > 0) { + switch (bd->type) { + case BDT_RAM: + MemCpy(bd->RAM_dsk + blk << BLK_SIZE_BITS, buf, cnt << BLK_SIZE_BITS); + break; + case BDT_ISO_FILE_READ: + case BDT_ISO_FILE_WRITE: + FBlkWrite(bd->file_dsk, buf, blk, cnt); + break; + case BDT_ATA: + case BDT_ATAPI: + res = CallExtStr("ATAWBlks", dv, buf, blk, cnt); + break; + case BDT_VIRTIO_BLK: + res = VIOWBlks(dv, buf, blk, cnt); + break; + } + bd->last_time = tS; + if (bd->flags & BDF_READ_CACHE) + CallExtStr("DskCacheAdd", dv, buf, blk, cnt); + } + if (unlock) + DrvUnlock(dv); + } catch if (unlock) + DrvUnlock(dv); + return res; +} + +@patch_jmp_rel32(&BlkRead, &BlkRead2); +@patch_jmp_rel32(&BlkWrite, &BlkWrite2); + +// DskBlkDev2.HC + +CBlkDev* BlkDevChk2(CBlkDev* bd, Bool except = TRUE) +{ // Check for valid BlkDev. Throw exception. + if (bd->type == BDT_VIRTIO_BLK) + return bd; + if (!bd || bd->bd_signature != BD_SIGNATURE_VAL || !(BDT_NULL < bd->type < BDT_TYPES_NUM)) { + if (except) + throw('BlkDev'); + else + return NULL; + } else + return bd; +} + +@patch_jmp_rel32(&BlkDevChk, &BlkDevChk2); + +// DskDrv2.HC + +DefineLstLoad("ST_BLKDEV_TYPES2", + "NULL\0RAM\0ATA\0FILE_READ\0FILE_WRITE\0ATAPI\0NULL\0NULL\0NULL\0NULL\0VIRTIO\0"); + +U8 DrvTextAttrGet2(U8 drv_let = 0) +{ // Get color of drive. + U8* blkdev_text_attr2 = blkdev_text_attr; + U8* drv_text_attr2 = drv_text_attr; + I64 dta_size = 3; + drv_let = Let2Let(drv_let); + if (drv_let == 'A') + return BLACK << 4 | WHITE; + if ('A' <= drv_let <= 'Z') + return blkdev_text_attr2[Let2BlkDevType(drv_let)] << 4 | drv_text_attr2[drv_let % dta_size]; + else + return BLACK << 4 | WHITE; +} + +U0 DrvRep2() +{ // Drive report. + CDrv* dv; + CBlkDev* bd; + I64 ch, i, drv_let, attr; + U8* st; + "\nDefined Drives:\n"; + for (i = 0, dv = blkdev.drvs; i < DRVS_NUM; i++, dv++) { + if (dv->dv_signature == DRV_SIGNATURE_VAL) { + bd = dv->bd; + drv_let = Drv2Let(dv); + if (Bt(&dv->fs_type, FStf_DISABLE)) + ch = '-'; + else if (drv_let == blkdev.boot_drv_let) + ch = ':'; + else + ch = '+'; + attr = DrvTextAttrGet(drv_let); + "\dFG,%d\d\dBG,%d\d%C %-8Z %-10Z %04X %04X %02X\n", + attr & 15, attr >> 4, drv_let, dv->fs_type &FSG_TYPE_MASK, "ST_DRV_TYPES", + bd->type, "ST_BLKDEV_TYPES2", bd->base0, bd->base1, bd->unit; + if (st = DrvModelNum(drv_let)) { + "Model#:%s\n", st; + Free(st); + } + if (st = DrvSerialNum(drv_let)) { + "Serial#:%s\n", st; + Free(st); + } + if (bd->type == BDT_ISO_FILE_READ || bd->type == BDT_ISO_FILE_WRITE) + "File=\"%s\"\n", bd->file_dsk_name; + "%016X-%016X\n\dFG\d\dBG\d", dv->drv_offset, dv->drv_offset + dv->size - 1; + } + } + "Home Dir:\"%s\"\n", blkdev.home_dir; +} + +@patch_jmp_rel32(&DrvTextAttrGet, &DrvTextAttrGet2); +@patch_jmp_rel32(&DrvRep, &DrvRep2); + +VirtioBlkInit; +MountVirtioBlk; + +if (Let2Drv('A', 0) && !Let2Drv('A')->root_clus) { + "[virtio-blk] RedSea filesystem not initialized, formatting.\n"; + Fmt('A', , FALSE, FSt_REDSEA); + Cd("M:/System/"); +} + +"virtio-blk "; diff --git a/System/FFI/Base.HC b/System/FFI/Base.HC new file mode 100644 index 0000000..e013399 --- /dev/null +++ b/System/FFI/Base.HC @@ -0,0 +1,51 @@ +#define PUSH_SYSV_REGS \ + asm {PUSH RCX PUSH RDX PUSH RBX PUSH RBP PUSH RSI PUSH RDI PUSH R8 PUSH R9 PUSH \ + R10 PUSH R11 PUSH R12 PUSH R13 PUSH R14 PUSH R15} +#define POP_SYSV_REGS \ + p0 = p0; \ + p1 = p1; \ + p2 = p2; \ + p3 = p3; \ + p4 = p4; \ + p5 = p5; \ + asm {POP R15 POP R14 POP R13 POP R12 POP R11 POP R10 POP R9 POP R8 POP RDI POP \ + RSI POP RBP POP RBX POP RDX POP RCX} +#define GET_SYSV_ARGS \ + asm {PUSH R9 PUSH R8 PUSH RCX PUSH RDX PUSH RSI PUSH RDI} \ + I64 reg RDI p0; \ + I64 reg RSI p1; \ + I64 reg RDX p2; \ + I64 reg RCX p3; \ + I64 reg R8 p4; \ + I64 reg R9 p5; \ + asm {POP RDI POP RSI POP RDX POP RCX POP R8 POP R9} + +#define MOV_ANS_RAX asm { MOV[&ans], RAX } +#define MOV_PARAM0_RDI asm {MOV [¶m0], RDI} + +I64 param0; +I64 elf_argc; +U8** elf_argv; + +asm { +_ELF_CALL:: + PUSH RBP + MOV RBP,RSP + MOV RAX,U64 SF_ARG1[RBP] + MOV RDI,U64 SF_ARG2[RBP] + MOV RSI,U64 SF_ARG3[RBP] + TEST RAX,RAX + JZ @@05 + CALL RAX +@@05: POP RBP + RET1 8 +} + +U0 _main() +{ + MOV_PARAM0_RDI + CallInd(_ELF_CALL, param0, elf_argc, elf_argv); + UserTaskCont; +} + +U0 _exit() { UserTaskCont; } diff --git a/System/FFI/ELF64.HC b/System/FFI/ELF64.HC new file mode 100644 index 0000000..50e6315 --- /dev/null +++ b/System/FFI/ELF64.HC @@ -0,0 +1,301 @@ +#define EI_NIDENT 16 +#define EM_X86_64 0x3E +#define ET_EXEC 2 +#define ET_DYN 3 + +U0 @elf64_debug_print(U8 fmt, ...) +{ + // FIXME: Remove unnecessary debug_print statements and PrintErr for errors. + no_warn fmt, argc, argv; +} + +class Elf64_Ehdr { + U8 e_ident[EI_NIDENT]; /* Magic number and other info */ + U16 e_type; /* Object file type */ + U16 e_machine; /* Architecture */ + U32 e_version; /* Object file version */ + U64 e_entry; /* Entry point virtual address */ + U64 e_phoff; /* Program header table file offset */ + U64 e_shoff; /* Section header table file offset */ + U32 e_flags; /* Processor-specific flags */ + U16 e_ehsize; /* ELF header size in bytes */ + U16 e_phentsize; /* Program header table entry size */ + U16 e_phnum; /* Program header table entry count */ + U16 e_shentsize; /* Section header table entry size */ + U16 e_shnum; /* Section header table entry count */ + U16 e_shstrndx; /* Section header string table index */ +}; + +class Elf64_Shdr { + U32 sh_name; /* Section name (string tbl index) */ + U32 sh_type; /* Section type */ + U64 sh_flags; /* Section flags */ + U64 sh_addr; /* Section virtual addr at execution */ + U64 sh_offset; /* Section file offset */ + U64 sh_size; /* Section size in bytes */ + U32 sh_link; /* Link to another section */ + U32 sh_info; /* Additional section information */ + U64 sh_addralign; /* Section alignment */ + U64 sh_entsize; /* Entry size if section holds table */ +}; + +class Elf64_Sym { + U32 st_name; /* Symbol name (string tbl index) */ + U8 st_info; /* Symbol type and binding */ + U8 st_other; /* Symbol visibility */ + U16 st_shndx; /* Section index */ + U64 st_value; /* Symbol value */ + U64 st_size; /* Symbol size */ +}; + +class PLT_entry { + U8 pad[0x10]; +}; + +class RELA_entry { + U64 r_offset; + U64 r_info; + I64 r_addend; +}; + +class Elf { + union { + U8* u8; + Elf64_Ehdr* ehdr; + } I64 size; + U8* dynstr; + Elf64_Sym* dynsym; + PLT_entry* plt; + RELA_entry* rela_dyn; + RELA_entry* rela_plt; + Elf64_Sym* strtab; + Elf64_Sym* symtab; + I64 rela_dyn_size; + I64 rela_plt_size; + I64 strtab_size; + I64 symtab_size; +}; + +U0(*_start) +(); + +U0 unimplemented_symbol() +{ + I32 s = 0xDEADF00D; + PrintWarn("Unimplemented symbol: %s\n", s); + Dbg; + while (1) + Sleep(1); +} + +Bool is_valid_elf(Elf* elf) +{ + Bool res = TRUE; + if (MemCmp(elf->u8 + 1, "ELF", 3)) { + @elf64_debug_print("Invalid signature (not ELF).\n"); + res = FALSE; + } + if (elf->ehdr->e_type != ET_EXEC && elf->ehdr->e_type != ET_DYN) { + @elf64_debug_print("Invalid object file type.\n"); + res = FALSE; + } + if (elf->ehdr->e_machine != EM_X86_64) { + @elf64_debug_print("Invalid architecture.\n"); + res = FALSE; + } + return res; +} + +U0 process_elf_section_header_table(Elf* elf) +{ + Elf64_Shdr* shdr = elf->u8 + elf->ehdr->e_shoff; + Elf64_Shdr* shdr_shstrtab = shdr + elf->ehdr->e_shstrndx; + U8* shstrtab = elf->u8 + shdr_shstrtab->sh_offset; + I64 i = 0; + while (i < elf->ehdr->e_shnum) { + if (!StrCmp(shstrtab + shdr->sh_name, ".symtab")) { + @elf64_debug_print("found symtab at 0x%08x, size = %d\n", shdr->sh_offset, + shdr->sh_size); + elf->symtab = elf->u8 + shdr->sh_offset; + elf->symtab_size = shdr->sh_size; + } + if (!StrCmp(shstrtab + shdr->sh_name, ".strtab")) { + @elf64_debug_print("found strtab at 0x%08x, size = %d\n", shdr->sh_offset, + shdr->sh_size); + elf->strtab = elf->u8 + shdr->sh_offset; + elf->strtab_size = shdr->sh_size; + } + if (shdr->sh_addr) { + MemCpy(shdr->sh_addr, elf->u8 + shdr->sh_offset, shdr->sh_size); + if (!StrCmp(shstrtab + shdr->sh_name, ".dynstr")) + elf->dynstr = shdr->sh_addr; + if (!StrCmp(shstrtab + shdr->sh_name, ".dynsym")) + elf->dynsym = shdr->sh_addr; + if (!StrCmp(shstrtab + shdr->sh_name, ".plt")) + elf->plt = shdr->sh_addr; + if (!StrCmp(shstrtab + shdr->sh_name, ".rela.dyn")) { + elf->rela_dyn = shdr->sh_addr; + elf->rela_dyn_size = shdr->sh_size / shdr->sh_entsize; + } + if (!StrCmp(shstrtab + shdr->sh_name, ".rela.plt")) { + elf->rela_plt = shdr->sh_addr; + elf->rela_plt_size = shdr->sh_size / shdr->sh_entsize; + } + if (!StrCmp(shstrtab + shdr->sh_name, ".bss") || !StrCmp(shstrtab + shdr->sh_name, ".tbss")) { + MemSet(shdr->sh_addr, NULL, shdr->sh_size); + @elf64_debug_print( + "Zeroed out section '%s' at physical address 0x%06x, size = %d bytes\n", + shstrtab + shdr->sh_name, shdr->sh_addr, shdr->sh_size); + } else + @elf64_debug_print( + "MemCpy section '%s' to physical address 0x%06x, size = %d bytes\n", + shstrtab + shdr->sh_name, shdr->sh_addr, shdr->sh_size); + if (!StrCmp(shstrtab + shdr->sh_name, ".bss")) { + MemSet(shdr->sh_addr, NULL, shdr->sh_size); + @elf64_debug_print("MemSet section '%s' at physical address 0x%06x to NULL, " + "size = %d bytes\n", + shstrtab + shdr->sh_name, shdr->sh_addr, shdr->sh_size); + } + } + shdr++; + i++; + } +} + +U0 process_elf_rela_dyn_entries(Elf* elf) +{ + I64 i; + U8* entry_name; + RELA_entry* rela_dyn = elf->rela_dyn; + for (i = 0; i < elf->rela_dyn_size; i++) { + entry_name = elf->dynstr + elf->dynsym[(rela_dyn->r_info >> 32)].st_name; + @elf64_debug_print("rela_dyn->r_offset = %08x\n", rela_dyn->r_offset); + @elf64_debug_print("entry name = '%s'\n", entry_name); + if (!StrCmp(entry_name, "__libc_start_main")) { + *(rela_dyn->r_offset)(U64*) = &_main; + @elf64_debug_print("Set value for .rela.dyn entry '%s' to: &_main\n", + entry_name); + } + if (!StrCmp(entry_name, "stdin")) { + *(rela_dyn->r_offset)(U64*) = 0; + @elf64_debug_print("Set value for .rela.dyn entry '%s' to: %d\n", entry_name, 0); + } + if (!StrCmp(entry_name, "stdout")) { + *(rela_dyn->r_offset)(U64*) = 1; + @elf64_debug_print("Set value for .rela.dyn entry '%s' to: %d\n", entry_name, 1); + } + if (!StrCmp(entry_name, "stderr")) { + *(rela_dyn->r_offset)(U64*) = 2; + @elf64_debug_print("Set value for .rela.dyn entry '%s' to: %d\n", entry_name, 2); + } + rela_dyn++; + } +} + +CHashClass* get_symbol_hash_entry(U8* entry_name) +{ + I64 i; + CHashSrcSym* sym; + CHashTable* tbl = Fs->hash_table; + while (tbl) { + for (i = 0; i < tbl->mask; i++) { + sym = tbl->body[i]; + while (sym) { + if (sym->type == HTT_CLASS) + if (!StrCmp(sym->str, entry_name)) + return sym; + sym = sym->next; + } + } + tbl = tbl->next; + } + return NULL; +} + +U64 get_symbol_address(U8* entry_name) +{ + CHash* h = HashFind(entry_name, Fs->hash_table, Fs->hash_table->mask); + if (!h) + return NULL; + switch (h->type) { + case HTT_GLBL_VAR: + return h(CHashGlblVar*)->data_addr; + break; + case HTT_FUN: + return h(CHashFun*)->exe_addr; + break; + default: + return NULL; + break; + } + return NULL; +} + +U0 process_elf_rela_plt_entries(Elf* elf) +{ + I64 i; + U32 handler; + U32* patch; + U8* entry_name; + Bool symbol_exists; + PLT_entry* plt = elf->plt; + RELA_entry* rela_plt = elf->rela_plt; + plt++; + for (i = 0; i < elf->rela_plt_size; i++) { + symbol_exists = FALSE; + entry_name = elf->dynstr + elf->dynsym[(rela_plt->r_info >> 32)].st_name; + handler = MAlloc(sizeof(unimplemented_symbol), erythros_mem_task->code_heap); + MemCpy(handler, &unimplemented_symbol, sizeof(unimplemented_symbol)); + patch = handler + 0x0A; + *patch = entry_name; + @patch_jmp_rel32(plt, handler); + @patch_call_rel32(handler + 0x16, &PrintErr); + //@patch_call_rel32(handler + 0x21, &_exit); + if (!StrCmp(entry_name, "__libc_start_main")) { + symbol_exists = TRUE; + @patch_jmp_rel32(plt, &_main); + @elf64_debug_print("Set value for .rela.plt entry '%s' to &_main\n", entry_name); + } + if (get_symbol_address(entry_name)) { + symbol_exists = TRUE; + @patch_jmp_rel32(plt, get_symbol_address(entry_name)); + @elf64_debug_print("Set value for .rela.plt entry '%s' to &%s\n", entry_name, + entry_name); + } + if (!symbol_exists) + @elf64_debug_print( + "Set value for .rela.plt entry '%s' to &unimplemented_symbol\n", + entry_name); + rela_plt++; + plt++; + } +} + +U0 load_elf(...) +{ + if (argc < 1) { + PrintErr("Not enough arguments.\n"); + return; + } + if (!FileFind(argv[0])) { + PrintErr("File not found: %s\n", argv[0]); + return; + } + + Elf elf; + elf.u8 = FileRead(argv[0], &elf.size); + @elf64_debug_print("Load file '%s', size = %d bytes\n", argv[0], elf.size); + + if (!is_valid_elf(&elf)) { + PrintErr("File is not a valid ELF x86-64 executable.\n"); + return; + } + + process_elf_section_header_table(&elf); + process_elf_rela_dyn_entries(&elf); + process_elf_rela_plt_entries(&elf); + + _start = elf.ehdr->e_entry; + elf_argc = argc; + elf_argv = argv; +} \ No newline at end of file diff --git a/System/FFI/LibC.HC b/System/FFI/LibC.HC new file mode 100644 index 0000000..9c89a44 --- /dev/null +++ b/System/FFI/LibC.HC @@ -0,0 +1,324 @@ +#define stdin 0 +#define stdout 1 +#define stderr 2 + +U0 bcmp() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + MemCmp(p0, p1, p2); + POP_SYSV_REGS +} + +U0 calloc() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + CAlloc(p0 * p1, erythros_mem_task->code_heap); + POP_SYSV_REGS +} + +U0 free() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + Free(p0); + POP_SYSV_REGS +} + +I64 @isatty() +{ + return 0; +} + +U0 isatty() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + // Dbg; + @isatty; + POP_SYSV_REGS +} + +I64 @fwrite(U8* ptr, I64 size, I64 nmemb, U64 stream) +{ + U8* tmp; + switch (stream) { + case stdout: + case stderr: + tmp = CAlloc((size * nmemb) + 1, erythros_mem_task->code_heap); + MemCpy(tmp, ptr, (size * nmemb)); +#ifdef QEMU_RUN_TESTS + QemuDebugMsg(tmp); +#endif + DocPutS(adam_task->put_doc, tmp); + Free(tmp); + // if (!MemCmp(tmp, "VERIFICATION FAILED", 19)) + // Break; + break; + default: + break; + } + return size * nmemb; +} + +U0 fwrite() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + @fwrite(p0, p1, p2, p3); + POP_SYSV_REGS +} + +U64 @getentropy(U8* buffer, U64 length) +{ + I64 i; + for (i = 0; i < length; i++) + buffer[i] = RandU64; + return 0; +} + +U0 getentropy() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + @getentropy(p0, p1); + POP_SYSV_REGS +} + +U0 htonl() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + EndianU32(p0); + POP_SYSV_REGS +} + +U0 ntohl() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + EndianU32(p0); + POP_SYSV_REGS +} + +U0 htons() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + EndianU16(p0); + POP_SYSV_REGS +} + +U0 ntohs() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + EndianU16(p0); + POP_SYSV_REGS +} + +U64 @malloc(I64 size) +{ + U64 res = MAlloc(size, malloc_mem_task[malloc_current_mem_task % MALLOC_MEM_TASK_COUNT]->code_heap); + malloc_current_mem_task++; + return res; +} + +U0 malloc() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + @malloc(p0); + POP_SYSV_REGS +} + +U0 memcmp() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + MemCmp(p0, p1, p2); + POP_SYSV_REGS +} + +U0 memcpy() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + MemCpy(p0, p1, p2); + POP_SYSV_REGS +} + +U8* @memmove(U8* dest, U8* src, I64 n) +{ + I64 i; + U8* from = src; + U8* to = dest; + if (from == to || n == 0) + return dest; + if (to > from && to - from < n) { + /* to overlaps with from */ + /* */ + /* */ + /* copy in reverse, to avoid overwriting from */ + for (i = n - 1; i >= 0; i--) + to[i] = from[i]; + return dest; + } + if (from > to && from - to < n) { + /* to overlaps with from */ + /* */ + /* */ + /* copy forwards, to avoid overwriting from */ + for (i = 0; i < n; i++) + to[i] = from[i]; + return dest; + } + MemCpy(dest, src, n); + return dest; +} + +U0 memmove() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + @memmove(p0, p1, p2); + POP_SYSV_REGS +} + +U0 memset() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + MemSet(p0, p1, p2); + POP_SYSV_REGS +} + +U0 putc() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + PutChars(p0); + POP_SYSV_REGS +} + +U0 rand() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + RandU64; + POP_SYSV_REGS +} + +U8* @realloc(U8* ptr, I64 size) +{ + U8* new; + if (!ptr) { + new = MAlloc(size, erythros_mem_task->code_heap); + } else { + new = MAlloc(size, erythros_mem_task->code_heap); + MemCpy(new, ptr, size); + Free(ptr); + } + return new; +} + +U0 realloc() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + @realloc(p0, p1); + POP_SYSV_REGS +} + +// FIXME: It is non-obvious how to take a [u8] and convert it to a +// formatted string in Jakt, so we have to do this hack for +// now. Hopefully, this will change soon. +U0 sprintf() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + StrPrint(p0, p1, p2, p3, p4, p5); + POP_SYSV_REGS +} + +I64 @strncmp(U8* s1, U8* s2, I32 n) +{ + U64 u1, u2; + + while (n-- > 0) { + u1 = *s1++; + u2 = *s2++; + u1 = u1 & 0xff; + u2 = u2 & 0xff; + if (u1 != u2) + return u1 - u2; + if (u1 == '\0') + return 0; + } + return 0; +} + +U0 strncmp() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + @strncmp(p0, p1, p2); + POP_SYSV_REGS +} + +U0 strcmp() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + StrCmp(p0, p1); + POP_SYSV_REGS +} + +U0 strlen() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + StrLen(p0); + POP_SYSV_REGS +} + +I64 tos_nist_offset = 5020; +#define NIST_TIME_OFFSET (tos_nist_offset - local_time_offset / CDATE_FREQ) + +public +I64 CDate2Unix(CDate dt) +{ // TempleOS datetime to Unix timestamp. + return ToI64((dt - Str2Date("1/1/1970")) / CDATE_FREQ + NIST_TIME_OFFSET); +} + +I64 @time(I64* ptr) +{ + no_warn ptr; + return CDate2Unix(Now); +} + +U0 time() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + @time(p0); + POP_SYSV_REGS +} + +U0 toupper() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + ToUpper(p0); + POP_SYSV_REGS +} + +U0 __assert_fail() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + "%s:%d: %s: %s\n", p1, p2, p3, p0; + Break; + POP_SYSV_REGS +} diff --git a/System/FFI/New.HC b/System/FFI/New.HC new file mode 100644 index 0000000..42d4e13 --- /dev/null +++ b/System/FFI/New.HC @@ -0,0 +1,35 @@ +U0 _ZdlPv() +{ + // operator delete(void*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + Free(p0); + POP_SYSV_REGS +} + +U0 _ZdlPvm() +{ + // operator delete(void*, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + Free(p0); + POP_SYSV_REGS +} + +U0 _Znwm() +{ + // operator new(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + MAlloc(p0, erythros_mem_task); + POP_SYSV_REGS +} + +U0 _ZnwmRKSt9nothrow_t() +{ + // operator new(unsigned long, std::nothrow_t const&) + PUSH_SYSV_REGS + GET_SYSV_ARGS + MAlloc(p0, erythros_mem_task); + POP_SYSV_REGS +} diff --git a/System/Jakt/DC.HC b/System/Jakt/DC.HC new file mode 100644 index 0000000..ea20b52 --- /dev/null +++ b/System/Jakt/DC.HC @@ -0,0 +1,288 @@ +U0 _Z8dc_aliasm() +{ + // dc_alias(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + DCAlias(p0); + POP_SYSV_REGS +} + +U0 _Z7dc_blotmmmm() +{ + // dc_blot(unsigned long, unsigned long, unsigned long, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GrBlot(p0, p1, p2, p3); + POP_SYSV_REGS +} + +U8* @dc_buffer(CDC* dc) { return dc->body; } + +U0 _Z9dc_bufferm() +{ + // dc_buffer(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_buffer(p0); + POP_SYSV_REGS +} + +I64 @dc_color(CDC* dc) { return dc->color; } + +U0 _Z8dc_colorm() +{ + // dc_color(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_color(p0); + POP_SYSV_REGS +} + +U0 @dc_copy(CDC* dest, I64 x, I64 y, CDC* src) +{ + + // If position is off-screen, return + if (x > dest->width - 1 || y > dest->height - 1) + return; + + // If device context dimensions match, MemCpy and return + if (dest->width_internal == src->width_internal && dest->height == src->height) { + MemCpy(dest->body, src->body, dest->width_internal * dest->height); + return; + } + + CDC* dc1 = DCAlias(dest); + CDC* dc2 = DCAlias(src); + + I64 src_line = 0; + I64 src_row = 0; + I64 clip_y = 0; + + // Handle horizontal clipping left + while (x < 0) { + dc2->x0++; + x++; + } + + // Handle vertical clipping top + while (y < 0) { + dc2->body += src->width_internal; + dc2->y0++; + y++; + } + + // default, clip line to copy as width-left off screen + src_line = src->width - dc2->x0; + + if (-dc2->x0 + x + src->width >= dest->width) { + src_line -= ((-dc2->x0 + x + src->width) - dest->width); + } + + dc2->body += dc2->x0; + clip_y = y; + + while (src_row < (src->height - dc2->y0) && clip_y < dest->height) { + MemCpy(dc1->body + (y * dest->width) + x, dc2->body, src_line); + dc2->body += src->width_internal; + dc1->body += dest->width_internal; + clip_y++; + src_row++; + } + + Free(dc2); + Free(dc1); +} + +U0 _Z7dc_copymmmm() +{ + // dc_copy(unsigned long, unsigned long, unsigned long, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_copy(p0, p1, p2, p3); + POP_SYSV_REGS +} + +U0 _Z10dc_destroym() +{ + // dc_destroy(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + DCDel(p0); + POP_SYSV_REGS +} + +U0 _Z14dc_draw_circlemlll() +{ + // dc_draw_circle(unsigned long, long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GrCircle3(p0, p1, p2, 0, p3); + POP_SYSV_REGS +} + +U0 _Z19dc_draw_filled_rectmllll() +{ + // dc_draw_filled_rect(unsigned long, long, long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GrRect(p0, p1, p2, p3, p4); + POP_SYSV_REGS +} + +U0 _Z12dc_draw_linemllll() +{ + // dc_draw_line(unsigned long, long, long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GrLine3(p0, p1, p2, 0, p3, p4, 0); + POP_SYSV_REGS +} + +U0 _Z13dc_draw_pixelmll() +{ + // dc_draw_pixel(unsigned long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GrPlot(p0, p1, p2); + POP_SYSV_REGS +} + +U0 _Z7dc_fillmm() +{ + // dc_fill(unsigned long, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + DCFill(p0, p1); + POP_SYSV_REGS +} + +CDC* @dc_gr_dc() { return gr.dc; } + +U0 _Z8dc_gr_dcv() +{ + // dc_gr_dc() + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_gr_dc(); + POP_SYSV_REGS +} + +I64 @dc_height(CDC* dc) { return dc->height; } + +U0 _Z9dc_heightm() +{ + // dc_height(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_height(p0); + POP_SYSV_REGS +} + +U0 _Z17dc_load_from_filePKc() +{ + // dc_load_from_file(char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GRRead(p0); + POP_SYSV_REGS +} + +U0 _Z6dc_newmm() +{ + // dc_new(unsigned long, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + DCNew(p0, p1); + POP_SYSV_REGS +} + +U0 _Z11dc_pixel_atmll() +{ + // dc_pixel_at(unsigned long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GrPeek(p0, p1, p2); + POP_SYSV_REGS +} + +U0 _Z16dc_replace_colormmm() +{ + // dc_replace_color(unsigned long, unsigned long, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + DCColorChg(p0, p1, p2); + POP_SYSV_REGS +} + +U0 _Z13dc_screenshotv() +{ + // dc_screenshot() + PUSH_SYSV_REGS + GET_SYSV_ARGS + DCScrnCapture(1); + POP_SYSV_REGS +} + +U0 _Z15dc_save_to_filePKcm() +{ + // dc_save_to_file(char const*, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GRWrite(p0, p1); + POP_SYSV_REGS +} + +U0 @dc_set_color(CDC* dc, I64 color) { dc->color = color; } + +U0 _Z12dc_set_colorml() +{ + // dc_set_color(unsigned long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_set_color(p0, p1); + POP_SYSV_REGS +} + +U0 @dc_set_thickness(CDC* dc, I64 thickness) { dc->thick = thickness; } + +U0 _Z16dc_set_thicknessml() +{ + // dc_set_thickness(unsigned long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_set_thickness(p0, p1); + POP_SYSV_REGS +} + +I64 @dc_thickness(CDC* dc) { return dc->thick; } + +U0 _Z12dc_thicknessm() +{ + // dc_thickness(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_thickness(p0); + POP_SYSV_REGS +} + +I64 @dc_width(CDC* dc) { return dc->width; } + +U0 _Z8dc_widthm() +{ + // dc_width(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_width(p0); + POP_SYSV_REGS +} + +I64 @dc_width_internal(CDC* dc) { return dc->width_internal; } + +U0 _Z17dc_width_internalm() +{ + // dc_width_internal(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @dc_width_internal(p0); + POP_SYSV_REGS +} \ No newline at end of file diff --git a/System/Jakt/IOPort.HC b/System/Jakt/IOPort.HC new file mode 100644 index 0000000..e192c14 --- /dev/null +++ b/System/Jakt/IOPort.HC @@ -0,0 +1,53 @@ +U0 _Z14ioport_read_u8t() +{ + // ioport_read_u8(unsigned short) + PUSH_SYSV_REGS + GET_SYSV_ARGS + InU8(p0); + POP_SYSV_REGS +} + +U0 _Z15ioport_read_u16t() +{ + // ioport_read_u16(unsigned short) + PUSH_SYSV_REGS + GET_SYSV_ARGS + InU16(p0); + POP_SYSV_REGS +} + +U0 _Z15ioport_read_u32t() +{ + // ioport_read_u32(unsigned short) + PUSH_SYSV_REGS + GET_SYSV_ARGS + InU32(p0); + POP_SYSV_REGS +} + +U0 _Z15ioport_write_u8th() +{ + // ioport_write_u8(unsigned short, unsigned char) + PUSH_SYSV_REGS + GET_SYSV_ARGS + OutU8(p0, p1); + POP_SYSV_REGS +} + +U0 _Z16ioport_write_u16tt() +{ + // ioport_write_u16(unsigned short, unsigned short) + PUSH_SYSV_REGS + GET_SYSV_ARGS + OutU16(p0, p1); + POP_SYSV_REGS +} + +U0 _Z16ioport_write_u32tj() +{ + // ioport_write_u32(unsigned short, unsigned int) + PUSH_SYSV_REGS + GET_SYSV_ARGS + OutU32(p0, p1); + POP_SYSV_REGS +} diff --git a/System/Jakt/Input.HC b/System/Jakt/Input.HC new file mode 100644 index 0000000..b427a40 --- /dev/null +++ b/System/Jakt/Input.HC @@ -0,0 +1,72 @@ +U0 _Z16input_get_stringPKc() +{ + // input_get_string(char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + GetStr(p0); + POP_SYSV_REGS +} + +Bool @input_key_down(U8 scancode) { return Bt(kbd.down_bitmap, scancode); } + +U0 _Z14input_key_downh() +{ + // input_key_down(unsigned char) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @input_key_down(p0); + POP_SYSV_REGS +} + +Bool @input_mouse_left() { return ms.lb; } + +U0 _Z16input_mouse_leftv() +{ + // input_mouse_left() + PUSH_SYSV_REGS + GET_SYSV_ARGS + @input_mouse_left(); + POP_SYSV_REGS +} + +Bool @input_mouse_right() { return ms.rb; } + +U0 _Z17input_mouse_rightv() +{ + // input_mouse_right() + PUSH_SYSV_REGS + GET_SYSV_ARGS + @input_mouse_right(); + POP_SYSV_REGS +} + +I64 @input_mouse_x() { return ms.pos.x; } + +U0 _Z13input_mouse_xv() +{ + // input_mouse_x() + PUSH_SYSV_REGS + GET_SYSV_ARGS + @input_mouse_x(); + POP_SYSV_REGS +} + +I64 @input_mouse_y() { return ms.pos.y; } + +U0 _Z13input_mouse_yv() +{ + // input_mouse_y() + PUSH_SYSV_REGS + GET_SYSV_ARGS + @input_mouse_y(); + POP_SYSV_REGS +} + +U0 _Z17input_press_a_keyv() +{ + // input_press_a_key() + PUSH_SYSV_REGS + GET_SYSV_ARGS + PressAKey; + POP_SYSV_REGS +} \ No newline at end of file diff --git a/System/Jakt/OS.HC b/System/Jakt/OS.HC new file mode 100644 index 0000000..88cd0cc --- /dev/null +++ b/System/Jakt/OS.HC @@ -0,0 +1,289 @@ +U0 _Z8os_blinkPKc() +{ + // os_blink(char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + F64 frequency = Str2F64(p0); + Print("called os_blink(%.1f)\n", frequency); + Blink(frequency); + POP_SYSV_REGS +} + +U64 @os_call(U8* function_name, U64 arg) +{ + if (!function_name) + return NULL; + if (!StrLen(function_name)) + return NULL; + CHash* h = HashFind(function_name, Fs->hash_table, Fs->hash_table->mask); + if (!h) + return NULL; + if (h->type & HTT_FUN == HTT_FUN) { + CallInd(h(CHashFun*)->exe_addr, arg); + } else { + return NULL; + } +} + +U0 _Z7os_callmm() +{ + // os_call(unsigned long, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @os_call(p0, p1); + POP_SYSV_REGS +} + +U0 _Z16os_device_callocj() +{ + // os_device_calloc(unsigned int) + PUSH_SYSV_REGS + GET_SYSV_ARGS + CAllocAligned(p0, 4096, erythros_mem_task->code_heap); + POP_SYSV_REGS +} + +U0 _Z7os_exitv() +{ + // os_exit() + PUSH_SYSV_REGS + GET_SYSV_ARGS + UserTaskCont; + POP_SYSV_REGS +} + +U8* @os_file_picker(U8* path, U8* glob) +{ + U8* full_path = CAlloc(StrLen(path) + StrLen(glob) + 4, erythros_mem_task); + CatPrint(full_path, "%s/%s", path, glob); + + CDirEntry* de = FilesFind(full_path); + Free(full_path); + + CDirEntry* tmpde; + U8* file_list = NULL; + U8* selected_file = NULL; + I64 list_pos = 0; + I64 list_size = 0; + + tmpde = de; + while (tmpde) { + list_size += StrLen(tmpde->name) + 2; + tmpde = tmpde->next; + } + + file_list = CAlloc(list_size, erythros_mem_task); + + tmpde = de; + while (tmpde) { + StrCpy(file_list + list_pos, tmpde->name); + list_pos += StrLen(tmpde->name) + 1; + tmpde = tmpde->next; + } + + I64 list_index = Adam("PopUpPickLst(0x%08x);\n", file_list); + Free(file_list); + list_pos = 0; + + if (list_index < 0) { + DirTreeDel(de); + return StrNew("", erythros_mem_task); + } + + tmpde = de; + while (tmpde) { + if (list_index == list_pos) { + selected_file = CAlloc(StrLen(path) + StrLen(tmpde->name) + 4, erythros_mem_task); + CatPrint(selected_file, "%s/%s", path, tmpde->name); + break; + } + StrCpy(file_list + list_pos, tmpde->name); + list_pos++; + tmpde = tmpde->next; + } + + DirTreeDel(de); + return selected_file; +} + +U0 _Z14os_file_pickerPKcS0_() +{ + // os_file_picker(char const*, char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @os_file_picker(p0, p1); + POP_SYSV_REGS +} + +U8* @os_files_list(U8* path) +{ + U8* full_path = CAlloc(StrLen(path) + 4, erythros_mem_task); + CatPrint(full_path, "%s", path); + + CDirEntry* de = FilesFind(full_path); + Free(full_path); + + CDateStruct ds; + CDirEntry* tmpde; + U8* file_list = NULL; + I64 list_size = 0; + + tmpde = de; + while (tmpde) { + list_size += StrLen(tmpde->name) + 48; // Should be enough for filename, date, + // filesize + semicolon separators + tmpde = tmpde->next; + } + + if (!list_size) + return NULL; + + file_list = CAlloc(list_size, erythros_mem_task); + + tmpde = de; + I64 counter = 0; + + while (tmpde) { + if (counter > 0) { + StrCpy(file_list + StrLen(file_list), "|"); + } + StrCpy(file_list + StrLen(file_list), tmpde->name); + if (tmpde->attr & RS_ATTR_DIR) + StrCpy(file_list + StrLen(file_list), "/"); + StrCpy(file_list + StrLen(file_list), ";"); + Date2Struct(&ds, tmpde->datetime); + StrPrint(file_list + StrLen(file_list), "%04d-%02d-%02d %02d:%02d", ds.year, + ds.mon, ds.day_of_mon, ds.hour, ds.min); + StrCpy(file_list + StrLen(file_list), ";"); + StrPrint(file_list + StrLen(file_list), "%d", tmpde->size); + tmpde = tmpde->next; + counter++; + } + + DirTreeDel(de); + return file_list; +} + +U0 _Z14os_path_existsPKc() +{ + // os_path_exists(char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + FileFind(p0); + POP_SYSV_REGS +} + +U0 _Z13os_files_listPKc() +{ + // os_files_list(char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @os_files_list(p0); + POP_SYSV_REGS +} + +Bool @os_is_vm() +{ + CRAXRBCRCXRDX res; + CPUId(0x40000000, &res); + if (res.rbx == 0x4B4D564B) + return TRUE; + return FALSE; +} + +U0 _Z8os_is_vmv() +{ + PUSH_SYSV_REGS + GET_SYSV_ARGS + @os_is_vm; + POP_SYSV_REGS +} + +U0 @os_pc_speaker(F64 frequency) +{ + I64 period; + if (!frequency) + OutU8(0x61, InU8(0x61) & ~3); + else { + period = ClampI64(SYS_TIMER_FREQ / frequency, 1, U16_MAX); + OutU8(0x43, 0xB6); + OutU8(0x42, period); + OutU8(0x42, period.u8[1]); + OutU8(0x61, 3 | InU8(0x61)); + } +} + +U0 _Z13os_pc_speakerPKc() +{ + // os_pc_speaker(char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + F64 frequency = Str2F64(p0); + @os_pc_speaker(frequency); + POP_SYSV_REGS +} + +U0 _Z9os_randomv() +{ + // os_random() + PUSH_SYSV_REGS + GET_SYSV_ARGS + RandU64; + POP_SYSV_REGS +} + +U0 _Z19os_read_entire_filePKcPl() +{ + // os_read_entire_file(char const*, long*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + FileRead(p0, p1); + POP_SYSV_REGS +} + +U0 @os_screenshot() +{ + CDC* dc = DCScrnCapture(, erythros_mem_task); + // Image.Write("B:/screenshot.png", dc); + DCDel(dc); +} + +U0 _Z13os_screenshotv() +{ + // os_screenshot() + PUSH_SYSV_REGS + GET_SYSV_ARGS + @os_screenshot; + POP_SYSV_REGS +} + +U8* @os_to_uppercase(U8* instr) +{ + if (!instr) + return NULL; + if (!StrLen(instr)) + return NULL; + U8* outstr = CAlloc(StrLen(instr) + 1, erythros_mem_task); + I64 i; + for (i = 0; i < StrLen(instr); i++) + outstr[i] = ToUpper(instr[i]); + return outstr; +} + +U0 _Z15os_to_uppercasePKc() +{ + // os_to_uppercase(char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @os_to_uppercase(p0); + POP_SYSV_REGS +} + +U0 _Z20os_write_entire_filePKcPhl() +{ + // os_write_entire_file(char const*, unsigned char*, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + FileWrite(p0, p1, p2); + POP_SYSV_REGS +} diff --git a/System/Jakt/PCI.HC b/System/Jakt/PCI.HC new file mode 100644 index 0000000..1a37903 --- /dev/null +++ b/System/Jakt/PCI.HC @@ -0,0 +1,62 @@ +U0 _Z8pci_findl() +{ + // pci_find(long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + PCIClassFind(p0, 0); + POP_SYSV_REGS +} + +U0 _Z11pci_read_u8llll() +{ + // pci_read_u8(long, long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + PCIReadU8(p0, p1, p2, p3); + POP_SYSV_REGS +} + +U0 _Z12pci_read_u16llll() +{ + // pci_read_u16(long, long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + PCIReadU16(p0, p1, p2, p3); + POP_SYSV_REGS +} + +U0 _Z12pci_read_u32llll() +{ + // pci_read_u32(long, long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + PCIReadU32(p0, p1, p2, p3); + POP_SYSV_REGS +} + +U0 _Z12pci_write_u8llllh() +{ + // pci_write_u8(long, long, long, long, unsigned char) + PUSH_SYSV_REGS + GET_SYSV_ARGS + PCIWriteU8(p0, p1, p2, p3, p4); + POP_SYSV_REGS +} + +U0 _Z13pci_write_u16llllt() +{ + // pci_write_u16(long, long, long, long, unsigned short) + PUSH_SYSV_REGS + GET_SYSV_ARGS + PCIWriteU16(p0, p1, p2, p3, p4); + POP_SYSV_REGS +} + +U0 _Z13pci_write_u32llllj() +{ + // pci_write_u32(long, long, long, long, unsigned int) + PUSH_SYSV_REGS + GET_SYSV_ARGS + PCIWriteU32(p0, p1, p2, p3, p4); + POP_SYSV_REGS +} diff --git a/System/Jakt/Time.HC b/System/Jakt/Time.HC new file mode 100644 index 0000000..f6f3f31 --- /dev/null +++ b/System/Jakt/Time.HC @@ -0,0 +1,37 @@ +U0 _Z9time_busyl() +{ + // time_busy(long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + Busy(p0); + POP_SYSV_REGS +} + +I64 @time_jiffies() { return cnts.jiffies; } + +U0 _Z12time_jiffiesv() +{ + // time_jiffies() + PUSH_SYSV_REGS + GET_SYSV_ARGS + @time_jiffies; + POP_SYSV_REGS +} + +U0 _Z8time_nowv() +{ + // time_now() + PUSH_SYSV_REGS + GET_SYSV_ARGS + Now; + POP_SYSV_REGS +} + +U0 _Z10time_sleepl() +{ + // time_sleep(long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + Sleep(p0); + POP_SYSV_REGS +} \ No newline at end of file diff --git a/System/Jakt/Window.HC b/System/Jakt/Window.HC new file mode 100644 index 0000000..7e91097 --- /dev/null +++ b/System/Jakt/Window.HC @@ -0,0 +1,94 @@ +U0 @window_draw_it(CTask* task, CDC* dc) +{ + if (task->user_data) + @dc_copy(dc, task->pix_left, task->pix_top, task->user_data); +} + +CTask* @window_user() +{ + CTask* task = Spawn(&UserCmdLine, , , 0); + TaskWait(task); + XTalk(task, + "while (1) { StrCpy(Fs->task_title, Fs->task_name); Sleep(1); };\n"); + return task; +} + +CTask* @window_create() +{ + CTask* task = @window_user; + task->draw_it = &@window_draw_it; + return task; +} + +U0 _Z13window_createv() +{ + // window_create() + PUSH_SYSV_REGS + GET_SYSV_ARGS + @window_create(); + POP_SYSV_REGS +} + +U0 _Z14window_destroym() +{ + // window_destroy(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + Kill(p0); + POP_SYSV_REGS +} + +Bool @window_is_focused(CTask* task) { return task == sys_focus_task; } + +U0 _Z17window_is_focusedm() +{ + // window_is_focused(unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @window_is_focused(p0); + POP_SYSV_REGS +} + +U0 @window_set_coordinates(CTask* task, I64 top, I64 left, I64 bottom, + I64 right) +{ + task->win_top = top; + task->win_left = left; + task->win_bottom = bottom; + task->win_right = right; +} + +U0 _Z22window_set_coordinatesmllll() +{ + // window_set_coordinates(unsigned long, long, long, long, long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @window_set_coordinates(p0, p1, p2, p3, p4); + POP_SYSV_REGS +} + +U0 @window_set_context(CTask* task, CDC* dc) { task->user_data = dc; } + +U0 _Z18window_set_contextmm() +{ + // window_set_context(unsigned long, unsigned long) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @window_set_context(p0, p1); + POP_SYSV_REGS +} + +U0 @window_set_title(CTask* task, U8* title) +{ + StrCpy(task->task_name, title); + StrCpy(task->task_title, title); +} + +U0 _Z16window_set_titlemPKc() +{ + // window_set_title(unsigned long, char const*) + PUSH_SYSV_REGS + GET_SYSV_ARGS + @window_set_title(p0, p1); + POP_SYSV_REGS +} diff --git a/System/Libraries/Animation2D.HC b/System/Libraries/Animation2D.HC new file mode 100644 index 0000000..9e2b0e6 --- /dev/null +++ b/System/Libraries/Animation2D.HC @@ -0,0 +1,65 @@ +class AnimationContext2D { + U64 signature; + I64 duration; // in jiffies + I64 index; // current frame + I64 length; // length in frames + I64 timer; // timer + cnts.jiffies = current ticks + Context2D** frame; // Context2D*, ... +}; + +AnimationContext2D* @animation2d_new_from_frames(Context2D** frames, I64 length, + I64 duration = 250) +{ + if (!frames || !length) + return NULL; + AnimationContext2D* actx = CAlloc(sizeof(AnimationContext2D)); + actx->signature = 'animated'; + actx->frame = frames; + actx->length = length; + actx->duration = duration; + return actx; +} + +Context2D* @animation2d_frame(AnimationContext2D* actx) +{ + I64 ticks = cnts.jiffies; + if (!actx) + return NULL; + if (!actx->length) + return NULL; + if (actx->index > actx->length - 1) + actx->index = 0; + if (!actx->index && !actx->timer) + actx->timer = ticks; + while (ticks >= actx->timer + actx->duration) { + actx->timer += actx->duration; + actx->index++; + if (actx->index > actx->length - 1) + actx->index = 0; + } + return actx->frame[actx->index]; +} + +Bool @animation2d_is_animation(AnimationContext2D* actx) +{ + return T(actx->signature == 'animated', TRUE, FALSE); +} + +U0 @animation2d_reset(AnimationContext2D* actx) { actx->index = 0; } + +class @animation2d +{ + Context2D* (*Frame)(AnimationContext2D* actx); + U0 (*Reset)(AnimationContext2D* actx); + Bool (*IsAnimation)(AnimationContext2D* actx); + AnimationContext2D* (*NewFromFrames)(Context2D** frames, I64 length, + I64 duration = 250); +}; + +@animation2d Animation2D; +Animation2D.IsAnimation = &@animation2d_is_animation; +Animation2D.NewFromFrames = &@animation2d_new_from_frames; +Animation2D.Frame = &@animation2d_frame; +Animation2D.Reset = &@animation2d_reset; + +"animation2d "; diff --git a/System/Libraries/Audio.HC b/System/Libraries/Audio.HC new file mode 100644 index 0000000..938f7ae --- /dev/null +++ b/System/Libraries/Audio.HC @@ -0,0 +1,143 @@ +// WAV header spec information: +// https://web.archive.org/web/20140327141505/https://ccrma.stanford.edu/courses/422/projects/WaveFormat/ +// http://www.topherlee.com/software/pcm-tut-wavformat.html + +class @sound_file_wav_header +{ + // RIFF Header + U8 riff_header[4]; // Contains "RIFF" + I32 wav_size; // Size of the wav portion of the file, which follows the first + // 8 bytes. File size - 8 + U8 wave_header[4]; // Contains "WAVE" + + // Format Header + U8 fmt_header[4]; // Contains "fmt " (includes trailing space) + I32 fmt_chunk_size; // Should be 16 for PCM + I16 audio_format; // Should be 1 for PCM. 3 for IEEE Float + I16 num_channels; + I32 sample_rate; + I32 byte_rate; // Number of bytes per second. sample_rate * num_channels * + // Bytes Per Sample + I16 sample_alignment; // num_channels * Bytes Per Sample + I16 bit_depth; // Number of bits per sample + + // Data + U8 data_header[4]; // Contains "data" + I32 data_bytes; // Number of bytes in data. Number of samples * num_channels * + // sample byte size + U8 bytes[0]; // Remainder of wave file is bytes +}; + +I64 @audio_get_available_output_stream() +{ + I64 stream = 0; + while (FifoI64Cnt(Audio.output[stream].data)) + stream++; + if (stream > AUDIO_MAX_STREAMS - 1) + return -1; + return stream; +} + +Bool @audio_buffer_is_wav(@sound_file_wav_header* wav, I64 size) +{ + if (!MemCmp(&wav->riff_header, "RIFF", 4) && !MemCmp(&wav->wave_header, "WAVE", 4)) + return TRUE; + return FALSE; +} + +U0 @audio_free_sound(Sound* snd) +{ + if (!snd) + return; + if (snd->data) + Free(snd->data); + Free(snd); +} + +I64 @audio_play_sound(Sound* snd) +{ + I64 i; + I64 stream = @audio_get_available_output_stream; + if (stream < 0) + return stream; + if (!snd->data || !snd->length) + return stream; + for (i = 0; i < snd->length; i++) + FifoI64Ins(Audio.output[stream].data, snd->data[i]); + return stream; +} + +Sound* @audio_sound_from_buffer(U32* buf, I64 length) +{ + if (!buf || !length) + return NULL; + Sound* snd = CAlloc(sizeof(Sound)); + snd->rate = 44100; + snd->channels = 2; + snd->bits = 16; + snd->data = buf; + snd->length = length; + return snd; +} + +U32* @audio_buffer_mono_to_stereo(U16* buf, I64 size) +{ + U32* out = CAlloc(size * 2); + I64 i; + for (i = 0; i < size / 2; i++) { + out[i].u16[0] = buf[i]; + out[i].u16[1] = buf[i]; + } + return out; +} + +U32* @audio_buffer_copy(U32* buf, I64 size) +{ + U32* out = MAlloc(size); + MemCpyU32(out, buf, size / 4); + return out; +} + +Sound* @audio_sound_from_file(U8* filename) +{ + if (!FileSystem.PathExists(filename)) + return NULL; + I64 length = 0; + U32* buf = NULL; + I64 size = 0; + U8* data = FileSystem.ReadFile(filename, &size); + if (!data) + return NULL; + if (@audio_buffer_is_wav(data, size)) { + @sound_file_wav_header* wav = data; + if (wav->fmt_chunk_size == 16 && wav->audio_format == 1 && wav->sample_rate == 48000) { + switch (wav->num_channels) { + case 1: + buf = @audio_buffer_mono_to_stereo(&wav->bytes, wav->data_bytes); + length = wav->data_bytes / 2; + break; + case 2: + buf = @audio_buffer_copy(&wav->bytes, wav->data_bytes); + length = wav->data_bytes / 4; + break; + } + } + } + Free(data); + return @audio_sound_from_buffer(buf, length); +} + +U0 @audio_snd(I8 ona = 0) { Audio.wavegen.frequency = Ona2Freq(ona); } + +Audio.SoundFromFile = &@audio_sound_from_file; +Audio.FreeSound = &@audio_free_sound; +Audio.PlaySound = &@audio_play_sound; + +Sound* @snd_beep = Audio.SoundFromFile("/mnt/redsea/t/Media/Sounds/Beep.wav"); + +U0 @audio_beep() { Audio.PlaySound(@snd_beep); } + +Audio.Beep = &@audio_beep; +Function.Patch(&Snd, &@audio_snd); + +"audio "; \ No newline at end of file diff --git a/System/Libraries/Base64.HC b/System/Libraries/Base64.HC new file mode 100644 index 0000000..e737197 --- /dev/null +++ b/System/Libraries/Base64.HC @@ -0,0 +1,90 @@ + +U8* @base64_chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; + +U8* @base64_decode(U8* input, I64* output_length) +{ + I64 input_length = StrLen(input); + if (input_length % 4 != 0) { + return NULL; // Invalid Base64 input length + } + + // Calculate the expected output length + *output_length = (3 * input_length) / 4; + if (input[input_length - 1] == '=') { + (*output_length)--; + } + if (input[input_length - 2] == '=') { + (*output_length)--; + } + + // Allocate memory for the decoded data + U8* decoded_data = CAlloc(*output_length, erythros_mem_task); + if (decoded_data == NULL) { + return NULL; // Memory allocation failed + } + + // Initialize variables for decoding process + I32 i, j = 0; + U32 sextet_bits = 0; + I64 sextet_count = 0; + U32 base64_value; + U8* char_pointer; + U8 input_find_buf[2]; + input_find_buf[1] = NULL; + + // Loop through the Base64 input and decode it + for (i = 0; i < input_length; i++) { + // Convert Base64 character to a 6-bit value + base64_value = 0; + if (input[i] == '=') { + base64_value = 0; + } else { + input_find_buf[0] = input[i]; + char_pointer = StrFirstOcc(@base64_chars, input_find_buf); + if (char_pointer == NULL) { + Free(decoded_data); + return NULL; // Invalid Base64 character + } + base64_value = char_pointer - @base64_chars; + } + + // Combine 6-bit values into a 24-bit sextet + sextet_bits = (sextet_bits << 6) | base64_value; + sextet_count++; + + // When a sextet is complete, decode it into three bytes + if (sextet_count == 4) { + decoded_data[j++] = (sextet_bits >> 16) & 0xFF; + decoded_data[j++] = (sextet_bits >> 8) & 0xFF; + decoded_data[j++] = sextet_bits & 0xFF; + sextet_bits = 0; + sextet_count = 0; + } + } + + return decoded_data; +} + +U8* @base64_encode(U8* input, I64 input_length) +{ + I64 i; + U8 buf[3]; + I64 c = 0; + U8* output = CAlloc(input_length * 2, erythros_mem_task); + + for (i = 0; i < input_length; i += 3) { + buf[0] = input[i]; + buf[1] = @t((i + 1 < input_length), input[i + 1], 0); + buf[2] = @t((i + 2 < input_length), input[i + 2], 0); + + output[c++] = @base64_chars[(buf[0] & 0xfc) >> 2]; + output[c++] = @base64_chars[((buf[0] & 0x03) << 4) + ((buf[1] & 0xf0) >> 4)]; + output[c++] = @t((i + 1 < input_length), @base64_chars[((buf[1] & 0x0f) << 2) + ((buf[2] & 0xc0) >> 6)], '='); + output[c++] = @t((i + 2 < input_length), @base64_chars[buf[2] & 0x3f], '='); + } + + output[c] = '\0'; + return output; +} + +"base64 "; diff --git a/System/Libraries/BitmapFont.HC b/System/Libraries/BitmapFont.HC new file mode 100644 index 0000000..35d8b68 --- /dev/null +++ b/System/Libraries/BitmapFont.HC @@ -0,0 +1,146 @@ +extern class Context2D; +extern U32 Color(I64 r, I64 g, I64 b, I64 a = 255); +extern U32 Peek2D(Context2D* ctx, I64 x, I64 y); + +class BitmapFont { + U8* name; + U8* char_map; + I64 line_height; + U16 bitmap[4096]; +}; + +class @bitmap_font_list +{ + @bitmap_font_list* prev; + @bitmap_font_list* next; + BitmapFont* font; +}; + +class @bitmapfont +{ + @bitmap_font_list* fonts; + U0 (*Add)(BitmapFont* font); + BitmapFont* (*GetByName)(U8* name); + U0 (*Init)(); +}; + +// BitmapFont* @bitmapfont_new_from_bdf_data(Context2D* ctx, U8* bdf_data) +//{ +// BitmapFont* font = CAlloc(sizeof(BitmapFont)); +// +// I64 bdf_lines_max = 0; +// U8** bdf_lines = String.Split(bdf_data, , &bdf_lines_max); +// +// I64 char_pos = 0; +// I64 char_x_pos = 0; +// I64 i, w; +// I64 xx, yy; +// /* +// while (*char_map++) { +// // Clear character bitmap +// for (i = 0; i < 16; i++) { +// font->bitmap[(char_pos * 16) + i] = 0; +// } +// // Get character width +// w = 0; +// for (xx = 0; xx < 16; xx++) { +// if (Peek2D(ctx, char_x_pos + xx, 0) == Color(255, 0, 0)) { +// w = xx; +// break; +// } +// } +// // Extract bitmap +// for (yy = 0; yy < 16; yy++) { +// for (xx = 0; xx < w + 1; xx++) { +// if (Peek2D(ctx, char_x_pos + xx, yy) == Color(0, 0, 0)) { +// font->bitmap[(char_pos * 16) + yy] |= 0x8000 >> xx; +// } +// } +// //"%016b\n", font->bitmap[(char_pos * 16) + yy]; +// } +// char_pos++; +// char_x_pos += w + 1; +// } +// */ +// return font; +// } + +BitmapFont* @bitmapfont_new_from_context2d(Context2D* ctx, U8* name, + U8* char_map, I64 fixed_width = 0) +{ + BitmapFont* font = CAlloc(sizeof(BitmapFont)); + font->name = StrNew(name); + font->char_map = StrNew(char_map); + + I64 char_pos = 0; + I64 char_x_pos = 0; + I64 i, w; + I64 xx, yy; + while (*char_map++) { + // Clear character bitmap + for (i = 0; i < 16; i++) { + font->bitmap[(char_pos * 16) + i] = 0; + } + w = fixed_width; + if (!w) { + // Get character width + for (xx = 0; xx < 16; xx++) { + if (Peek2D(ctx, char_x_pos + xx, 0) == Color(255, 0, 0)) { + w = xx; + break; + } + } + } + // Extract bitmap + for (yy = 0; yy < 16; yy++) { + for (xx = 0; xx < w + 1; xx++) { + if (Peek2D(ctx, char_x_pos + xx, yy) == Color(0, 0, 0)) { + font->bitmap[(char_pos * 16) + yy] |= 0x8000 >> xx; + } + } + //"%016b\n", font->bitmap[(char_pos * 16) + yy]; + } + char_pos++; + char_x_pos += w + 1; + } + return font; +} + +@bitmapfont BitmapFonts; + +U0 @bitmap_fonts_add(BitmapFont* font) +{ + @bitmap_font_list* fonts = BitmapFonts.fonts; + while (fonts->next) { + fonts = fonts->next; + } + @bitmap_font_list* font_list_item = CAlloc(sizeof(@bitmap_font_list)); + font_list_item->prev = fonts; + font_list_item->font = font; + fonts->next = font_list_item; +} + +BitmapFont* @bitmap_fonts_get_by_name(U8* name) +{ + @bitmap_font_list* fonts = BitmapFonts.fonts; + while (fonts) { + if (fonts->font) { + if (!StrCmp(fonts->font->name, name)) + return fonts->font; + } + fonts = fonts->next; + } + return NULL; +} + +U0 @bitmap_fonts_init() +{ + BitmapFonts.fonts = CAlloc(sizeof(@bitmap_font_list)); +} + +BitmapFonts.Add = &@bitmap_fonts_add; +BitmapFonts.GetByName = &@bitmap_fonts_get_by_name; +BitmapFonts.Init = &@bitmap_fonts_init; +BitmapFonts.Init(); + +"bitmapfont "; \ No newline at end of file diff --git a/System/Libraries/Clipboard.HC b/System/Libraries/Clipboard.HC new file mode 100644 index 0000000..c920a83 --- /dev/null +++ b/System/Libraries/Clipboard.HC @@ -0,0 +1,125 @@ +#define CLIP_MSG_NULL 0 +#define CLIP_MSG_INSERT 1 +#define CLIP_MSG_REMOVE 2 + +#define CLIP_TYPE_NULL 0 +#define CLIP_TYPE_TEXT 1 +#define CLIP_TYPE_DATA 2 + +class @clipboard_item +{ + I64 length; + I64 type; +}; + +class ClipboardTextItem : @clipboard_item { + U8* text; +} + +class ClipboardDataItem : @clipboard_item { + U8* data; +} + +class @clipboard_list_item +{ + @clipboard_list_item* prev; + @clipboard_list_item* next; + @clipboard_item* item; +}; + +class @clipboard +{ + CTask* task; + I64 length; + @clipboard_list_item* items; + U0 (*Init)(); + U0 (*Insert)(I64 type, U8* data); + I64 (*Length)(); + U0(*Task) + (); +}; + +@clipboard Clipboard; + +U0 @clipboard_add(@clipboard_item* item) +{ + @clipboard_list_item* items = Clipboard.items; + while (items->next) { + items = items->next; + } + @clipboard_list_item* new_item = CAlloc(sizeof(@clipboard_list_item)); + new_item->prev = items; + new_item->item = item; + items->next = new_item; + Clipboard.length++; + Clipboard.items->prev = new_item; +} + +U0 @clipboard_init() { Clipboard.items = CAlloc(sizeof(@clipboard_list_item)); } + +I64 @clipboard_length() { return Clipboard.length; } + +U0 @clipboard_ipc_queue_process() +{ + IpcMessage* msg; + msg = Ipc.MsgRecv(); + if (msg) { + switch (msg->type) { + case CLIP_MSG_INSERT: + @clipboard_add(msg->payload); + break; + case CLIP_MSG_REMOVE: + // FIXME: Handle this + break; + default: + break; + } + Free(msg); + } +} + +U0 @clipboard_insert_text(U8* text) +{ + IpcMessage* msg = CAlloc(sizeof(IpcMessage)); + ClipboardTextItem* item = CAlloc(sizeof(ClipboardTextItem)); + item->length = StrLen(text); + item->type = CLIP_TYPE_TEXT; + item->text = text; + msg->client = NULL; // FIXME: Do we care about client here? :/ + msg->type = CLIP_MSG_INSERT; + msg->payload = item; + System.Log(Fs, "Sent message → ClipInsert -> \"%s\"", text); + Ipc.MsgSend(Clipboard.task, msg); +} + +U0 @clipboard_insert(I64 type, U8* data) +{ + switch (type) { + case CLIP_TYPE_TEXT: + @clipboard_insert_text(data); + break; + case CLIP_TYPE_DATA: + // Reserved + break; + default: + break; + } +} + +U0 @clipboard_task() +{ + Ipc.InitQueue(Fs); + Clipboard.task = Fs; + System.Log(Fs, "Task running at 0x%08x", Fs); + while (1) { + @clipboard_ipc_queue_process(); + Sleep(1); + } +} + +Clipboard.Init = &@clipboard_init; +Clipboard.Insert = &@clipboard_insert; +Clipboard.Length = &@clipboard_length; +Clipboard.Task = &@clipboard_task; + +"clipboard "; diff --git a/System/Libraries/Display.HC b/System/Libraries/Display.HC new file mode 100644 index 0000000..df09f3b --- /dev/null +++ b/System/Libraries/Display.HC @@ -0,0 +1,49 @@ +Silent(ON); + +#define FB_NONE 0x00 +#define FB_VMSVGA 0x01 + +I64 @display_init(I64 width, I64 height, I64 bpp, I64 driver) +{ + I64 err; + Display.width = width; + Display.height = height; + Display.bpp = bpp; + Display.driver = driver; + Display.fb = NULL; + switch (Display.driver) { + case FB_VMSVGA: + err = VMSVGA.Init(Display.width, Display.height, Display.bpp); + if (err) + return err; + Display.fb = VMSVGA.FrameBuffer(); + Display.Update = &@vmsvga_display_update; + break; + default: + //"Unsupported display driver\n"; + return -1; + break; + } + text.cols = Display.width / 8; + text.rows = Display.height / 16; + text.raw_col = 0; + return 0; +} + +I64 @display_get_width() { return Display.width; } + +I64 @display_get_height() { return Display.height; } + +I64 @display_get_bpp() { return Display.bpp; } + +I64 @display_get_driver() { return Display.driver; } + +Display.Init = &@display_init; +Display.Width = &@display_get_width; +Display.Height = &@display_get_height; +Display.Bpp = &@display_get_bpp; +Display.Driver = &@display_get_driver; + +Silent(OFF); + +"display "; diff --git a/System/Libraries/FileSystem.HC b/System/Libraries/FileSystem.HC new file mode 100644 index 0000000..d35a1f7 --- /dev/null +++ b/System/Libraries/FileSystem.HC @@ -0,0 +1,234 @@ +#define FS_TYPE_UNSUPPORTED -1 +#define FS_TYPE_SYSTEM 0 +#define FS_TYPE_REDSEA 1 +#define FS_TYPE_9P 2 + +#define DE_TYPE_FILE 0 +#define DE_TYPE_DIR 1 + +class @dir_entry +{ + U8 mode; + U8 type; + U32 atime; + U32 mtime; + U64 size; + U8 name[255]; + U8 uid[255]; + U8 gid[255]; + @dir_entry* next; +}; + +extern Bool @plan9fs_file_find(U8* path); +extern @dir_entry* @plan9fs_get_files(U8* path); +extern U8* @plan9fs_read_file(U8* path, I64* size); +extern I64 @plan9fs_write_file(U8* path, U64 buffer, I64 size); + +class @filesystem +{ + I64 root_fs_type; + @dir_entry (*GetFiles)(U8* path); + U8* (*GetFileExtension)(U8* path); + U0 (*Init)(); + Bool (*PathExists)(U8* path); + U8* (*ReadFile)(U8* path, I64* size); + I64* (*WriteFile)(U8* path, U64 buffer, I64 size); +}; + +@filesystem FileSystem; + +U8* @filesystem_resolve_path(U8* path) +{ + U8* abs_path = CAlloc(StrLen(path)); + I64 argc; + I64 i; + I64 pos = 0; + U8** argv; + U8** outv; + U8* path_cpy = StrNew(path); + argv = String.Split(path_cpy, '/', &argc); + outv = CAlloc(sizeof(U64) * argc); + + for (i = 0; i < argc; i++) { + if (!(!StrCmp(argv[i], ".") || !StrCmp(argv[i], "") || !StrCmp(argv[i], ".."))) { + outv[pos] = argv[i]; + pos++; + } + if (!StrCmp(argv[i], "..")) { + pos = Max(0, pos - 1); + } + } + for (i = 0; i < pos; i++) + String.Append(abs_path, "/%s", outv[i]); + Free(path_cpy); + Free(outv); + if (abs_path[StrLen(abs_path) - 1] == '/') + abs_path[StrLen(abs_path) - 1] = NULL; + if (!StrLen(abs_path)) + StrCpy(abs_path, "/"); + return abs_path; +} + +I64 @filesystem_get_type(U8* path) +{ + if (!MemCmp(path, "/mnt/redsea/", 12) && StrLen(path) > 12) + return FS_TYPE_REDSEA; + if (!MemCmp(path, "/sys/", 5)) + return FS_TYPE_SYSTEM; + return FileSystem.root_fs_type; +} + +@dir_entry* @filesystem_get_files_9p(U8* path) +{ + return @plan9fs_get_files(path); +} + +@dir_entry* @filesystem_get_files_redsea(U8* path) +{ + CDirEntry* de = FilesFind(path); + CDirEntry* tmpde = NULL; + @dir_entry* entries = NULL; + @dir_entry* entry = NULL; + @dir_entry* new = NULL; + if (de) { + entries = CAlloc(sizeof(@dir_entry)); + entry = entries; + tmpde = de; + while (tmpde) { + new = CAlloc(sizeof(@dir_entry)); + entry->next = new; + + StrCpy(&new->name, &tmpde->name); + StrCpy(&new->uid, + "templeos"); // No file ownership in TempleOS + StrCpy(&new->gid, + "templeos"); // No file ownership in TempleOS + new->size = tmpde->size; + new->type = T(IsDir(tmpde->full_name), 1, 0); + + entry = new; + tmpde = tmpde->next; + } + DirTreeDel(de); + return entries; + } + return NULL; +} + +@dir_entry* @filesystem_get_files(U8* path) +{ + if (!path) + return NULL; + U8 buf[512]; + I64 type = @filesystem_get_type(path); + switch (type) { + case FS_TYPE_SYSTEM: + SysHlt; + break; + case FS_TYPE_REDSEA: + StrPrint(&buf, "%c:%s", ToUpper(path[12]), path + 13); + if (buf[StrLen(&buf) - 1] == ':') + buf[StrLen(&buf)] = '/'; + if (buf[StrLen(&buf) - 1] == '/') + buf[StrLen(&buf)] = '.'; + return @filesystem_get_files_redsea(&buf); + break; + case FS_TYPE_9P: + return @filesystem_get_files_9p(path); + break; + default: + break; + } + return NULL; +} + +U8* @filesystem_get_file_extension(U8* path) +{ + return (StrLastOcc(path, ".") + 1); +} + +Bool @filesystem_path_exists(U8* opath) +{ + if (!opath) + return FALSE; + U8 buf[512]; + U8* path = @filesystem_resolve_path(opath); + I64 type = @filesystem_get_type(path); + switch (type) { + case FS_TYPE_SYSTEM: + return NULL; + SysHlt; + break; + case FS_TYPE_REDSEA: + StrPrint(&buf, "%c:%s", ToUpper(path[12]), path + 13); + if (buf[StrLen(&buf) - 1] == ':') + buf[StrLen(&buf)] = '/'; + if (buf[StrLen(&buf) - 1] == '/') + buf[StrLen(&buf)] = '.'; + Free(path); + return FileFind(&buf); + break; + case FS_TYPE_9P: + return NULL; + Free(path); + return @plan9fs_file_find(path); + break; + default: + break; + } + return NULL; +} + +U8* @filesystem_read_file(U8* path, I64* size) +{ + if (!path) + return FALSE; + U8 buf[512]; + I64 type = @filesystem_get_type(path); + switch (type) { + case FS_TYPE_SYSTEM: + SysHlt; + break; + case FS_TYPE_REDSEA: + StrPrint(&buf, "%c:%s", ToUpper(path[12]), path + 13); + return FileRead(&buf, size); + break; + case FS_TYPE_9P: + return @plan9fs_read_file(path, size); + break; + default: + break; + } + return NULL; +} + +I64 @filesystem_write_file(U8* path, U8* buffer, I64 size) +{ + if (!path || !buffer || !size) + return FALSE; + U8 buf[512]; + I64 type = @filesystem_get_type(path); + switch (type) { + case FS_TYPE_SYSTEM: + SysHlt; + break; + case FS_TYPE_REDSEA: + StrPrint(&buf, "%c:%s", ToUpper(path[12]), path + 13); + return FileWrite(&buf, buffer, size); + break; + case FS_TYPE_9P: + return @plan9fs_write_file(path, buffer, size); + break; + default: + break; + } + return NULL; +} + +FileSystem.GetFiles = &@filesystem_get_files; +FileSystem.GetFileExtension = &@filesystem_get_file_extension; +FileSystem.PathExists = &@filesystem_path_exists; +FileSystem.ReadFile = &@filesystem_read_file; +FileSystem.WriteFile = &@filesystem_write_file; + +"filesystem "; \ No newline at end of file diff --git a/System/Libraries/Function.HC b/System/Libraries/Function.HC new file mode 100644 index 0000000..d66c443 --- /dev/null +++ b/System/Libraries/Function.HC @@ -0,0 +1,28 @@ +U0 @function_insert_call(U32 from, U32 to) +{ + *(from(U8*)) = 0xE8; + *((from + 1)(I32*)) = to - from - 5; +} + +U0 @function_patch(U32 from, U32 to) +{ + *(from(U8*)) = 0xE9; + *((from + 1)(I32*)) = to - from - 5; +} + +class @function +{ + U0(*InsertCall) + (U32 from, U32 to); + U0(*Patch) + (U32 from, U32 to); +}; + +@function Function; +Function.InsertCall = &@function_insert_call; +Function.Patch = &@function_patch; + +// usage: Function.InsertCall(addr, &func); +// usage: Function.Patch(&old_func, &new_func); + +"function "; \ No newline at end of file diff --git a/System/Libraries/Graphics2D.HC b/System/Libraries/Graphics2D.HC new file mode 100644 index 0000000..82aa7d0 --- /dev/null +++ b/System/Libraries/Graphics2D.HC @@ -0,0 +1,1428 @@ +class Context2D { + I64 width; + I64 height; + I64 bpp; + U32* fb; +}; + +class Bounds2D { + I64 x1; + I64 y1; + I64 x2; + I64 y2; +}; + +U32 Color(I64 r, I64 g, I64 b, I64 a = 255) +{ + U32 c; + c.u8[0] = b; + c.u8[1] = g; + c.u8[2] = r; + c.u8[3] = a; + return c; +} + +U32 ColorHSVToRGB(I64 hue, I64 saturation, I64 value) +{ + if (!(0 <= hue < 360)) + hue = 0; + if (!(0 <= saturation <= 100)) + saturation = 100; + if (!(0 <= value <= 100)) + value = 100; + + F64 s = saturation / 100.0; + F64 v = value / 100.0; + + F64 c = v * s; + F64 x = c * (1 - Abs(((hue / 60.0) % 2) - 1)); + F64 m = v - c; + + F64 r, g, b; + switch (hue) { + case 0...59: + r = c; + g = x; + b = 0; + break; + case 60...119: + r = x; + g = c; + b = 0; + break; + case 120...179: + r = 0; + g = c; + b = x; + break; + case 180...239: + r = 0; + g = x; + b = c; + break; + case 240...299: + r = x; + g = 0; + b = c; + break; + case 300...359: + r = c; + g = 0; + b = x; + break; + } + + return Color(ToI64((r + m) * 255), ToI64((g + m) * 255), ToI64((b + m) * 255)); +} + +U0 ColorRGBToHSV(U32 color, I64* hue, I64* saturation, I64* value) +{ + if (!hue || !saturation || !value) + return; + + F64 r = color.u8[2] / 255.0; + F64 g = color.u8[1] / 255.0; + F64 b = color.u8[0] / 255.0; + + F64 cMax = Max(r, Max(g, b)); + F64 cMin = Min(r, Min(g, b)); + + F64 delta = cMax - cMin; + + if (delta == 0) + *hue = 0; + else if (cMax == r) + *hue = 60 * (((g - b) / delta) % 6); + else if (cMax == g) + *hue = 60 * (((b - r) / delta) + 2); + else + *hue = 60 * (((r - g) / delta) + 4); + + if (cMax != 0) + *saturation = (delta / cMax) * 100; + else + *saturation = 0; + *value = cMax * 100; +} + +/* + * ISO Latin-1 Font + * + * Copyright (c) 2000 + * Ka-Ping Yee + * + * This font may be freely used for any purpose. + */ + +/* + * adjusted 'A' 'V' to improve their dense appearance (ie. lightened) + * adjusted 'i' 'l' to improve their flow within a word (ie. widened) + * adjusted 'E' 'F' '#' + */ + +U8 console_font[256 * 16] = { + /* 0 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 1 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 2 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 3 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 4 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 5 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 6 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 7 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 8 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 9 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 10 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 11 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 12 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 13 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 14 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 15 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 16 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 17 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 18 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 19 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 20 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 21 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 22 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 23 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 24 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 25 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 26 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 27 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 28 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 29 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 30 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 31 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 32 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 33 */ 0x00, 0x00, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x00, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 34 */ 0x00, 0x00, 0x6c, 0x6c, 0x36, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 35 */ 0x00, 0x00, 0x00, 0x36, 0x36, 0x7f, 0x36, 0x36, + 0x7f, 0x36, 0x36, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 36 */ 0x00, 0x08, 0x08, 0x3e, 0x6b, 0x0b, 0x0b, 0x3e, + 0x68, 0x68, 0x6b, 0x3e, 0x08, 0x08, 0x00, 0x00, + /* 37 */ 0x00, 0x00, 0x00, 0x33, 0x13, 0x18, 0x08, 0x0c, + 0x04, 0x06, 0x32, 0x33, 0x00, 0x00, 0x00, 0x00, + /* 38 */ 0x00, 0x00, 0x1c, 0x36, 0x36, 0x1c, 0x6c, 0x3e, + 0x33, 0x33, 0x7b, 0xce, 0x00, 0x00, 0x00, 0x00, + /* 39 */ 0x00, 0x00, 0x18, 0x18, 0x0c, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 40 */ 0x00, 0x00, 0x30, 0x18, 0x18, 0x0c, 0x0c, 0x0c, + 0x0c, 0x0c, 0x18, 0x18, 0x30, 0x00, 0x00, 0x00, + /* 41 */ 0x00, 0x00, 0x0c, 0x18, 0x18, 0x30, 0x30, 0x30, + 0x30, 0x30, 0x18, 0x18, 0x0c, 0x00, 0x00, 0x00, + /* 42 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x36, 0x1c, 0x7f, + 0x1c, 0x36, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 43 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0x18, 0x7e, + 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 44 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x18, 0x18, 0x0c, 0x00, 0x00, 0x00, + /* 45 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7e, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 46 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 47 */ 0x00, 0x00, 0x60, 0x20, 0x30, 0x10, 0x18, 0x08, + 0x0c, 0x04, 0x06, 0x02, 0x03, 0x00, 0x00, 0x00, + /* 48 */ 0x00, 0x00, 0x3e, 0x63, 0x63, 0x63, 0x6b, 0x6b, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 49 */ 0x00, 0x00, 0x18, 0x1e, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x18, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 50 */ 0x00, 0x00, 0x3e, 0x63, 0x60, 0x60, 0x30, 0x18, + 0x0c, 0x06, 0x03, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 51 */ 0x00, 0x00, 0x3e, 0x63, 0x60, 0x60, 0x3c, 0x60, + 0x60, 0x60, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 52 */ 0x00, 0x00, 0x30, 0x38, 0x3c, 0x36, 0x33, 0x7f, + 0x30, 0x30, 0x30, 0x30, 0x00, 0x00, 0x00, 0x00, + /* 53 */ 0x00, 0x00, 0x7f, 0x03, 0x03, 0x3f, 0x60, 0x60, + 0x60, 0x60, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 54 */ 0x00, 0x00, 0x3c, 0x06, 0x03, 0x03, 0x3f, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 55 */ 0x00, 0x00, 0x7f, 0x60, 0x30, 0x30, 0x18, 0x18, + 0x18, 0x0c, 0x0c, 0x0c, 0x00, 0x00, 0x00, 0x00, + /* 56 */ 0x00, 0x00, 0x3e, 0x63, 0x63, 0x63, 0x3e, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 57 */ 0x00, 0x00, 0x3e, 0x63, 0x63, 0x63, 0x7e, 0x60, + 0x60, 0x60, 0x30, 0x1e, 0x00, 0x00, 0x00, 0x00, + /* 58 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0x18, 0x00, + 0x00, 0x00, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 59 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0x18, 0x00, + 0x00, 0x00, 0x18, 0x18, 0x0c, 0x00, 0x00, 0x00, + /* 60 */ 0x00, 0x00, 0x60, 0x30, 0x18, 0x0c, 0x06, 0x06, + 0x0c, 0x18, 0x30, 0x60, 0x00, 0x00, 0x00, 0x00, + /* 61 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x7e, 0x00, 0x00, + 0x7e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 62 */ 0x00, 0x00, 0x06, 0x0c, 0x18, 0x30, 0x60, 0x60, + 0x30, 0x18, 0x0c, 0x06, 0x00, 0x00, 0x00, 0x00, + /* 63 */ 0x00, 0x00, 0x3e, 0x63, 0x60, 0x30, 0x30, 0x18, + 0x18, 0x00, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 64 */ 0x00, 0x00, 0x3c, 0x66, 0x73, 0x7b, 0x6b, 0x6b, + 0x7b, 0x33, 0x06, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 65 */ 0x00, 0x00, 0x3e, 0x63, 0x63, 0x63, 0x7f, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 66 */ 0x00, 0x00, 0x3f, 0x63, 0x63, 0x63, 0x3f, 0x63, + 0x63, 0x63, 0x63, 0x3f, 0x00, 0x00, 0x00, 0x00, + /* 67 */ 0x00, 0x00, 0x3c, 0x66, 0x03, 0x03, 0x03, 0x03, + 0x03, 0x03, 0x66, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 68 */ 0x00, 0x00, 0x1f, 0x33, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x33, 0x1f, 0x00, 0x00, 0x00, 0x00, + /* 69 */ 0x00, 0x00, 0x7f, 0x03, 0x03, 0x03, 0x3f, 0x03, + 0x03, 0x03, 0x03, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 70 */ 0x00, 0x00, 0x7f, 0x03, 0x03, 0x03, 0x3f, 0x03, + 0x03, 0x03, 0x03, 0x03, 0x00, 0x00, 0x00, 0x00, + /* 71 */ 0x00, 0x00, 0x3c, 0x66, 0x03, 0x03, 0x03, 0x73, + 0x63, 0x63, 0x66, 0x7c, 0x00, 0x00, 0x00, 0x00, + /* 72 */ 0x00, 0x00, 0x63, 0x63, 0x63, 0x63, 0x7f, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 73 */ 0x00, 0x00, 0x3c, 0x18, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x18, 0x18, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 74 */ 0x00, 0x00, 0x30, 0x30, 0x30, 0x30, 0x30, 0x30, + 0x30, 0x30, 0x33, 0x1e, 0x00, 0x00, 0x00, 0x00, + /* 75 */ 0x00, 0x00, 0x63, 0x33, 0x1b, 0x0f, 0x07, 0x07, + 0x0f, 0x1b, 0x33, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 76 */ 0x00, 0x00, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, + 0x03, 0x03, 0x03, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 77 */ 0x00, 0x00, 0x63, 0x63, 0x77, 0x7f, 0x7f, 0x6b, + 0x6b, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 78 */ 0x00, 0x00, 0x63, 0x63, 0x67, 0x6f, 0x6f, 0x7b, + 0x7b, 0x73, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 79 */ 0x00, 0x00, 0x3e, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 80 */ 0x00, 0x00, 0x3f, 0x63, 0x63, 0x63, 0x63, 0x3f, + 0x03, 0x03, 0x03, 0x03, 0x00, 0x00, 0x00, 0x00, + /* 81 */ 0x00, 0x00, 0x3e, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x6f, 0x7b, 0x3e, 0x30, 0x60, 0x00, 0x00, + /* 82 */ 0x00, 0x00, 0x3f, 0x63, 0x63, 0x63, 0x63, 0x3f, + 0x1b, 0x33, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 83 */ 0x00, 0x00, 0x3e, 0x63, 0x03, 0x03, 0x0e, 0x38, + 0x60, 0x60, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 84 */ 0x00, 0x00, 0x7e, 0x18, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x18, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 85 */ 0x00, 0x00, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 86 */ 0x00, 0x00, 0x63, 0x63, 0x63, 0x63, 0x63, 0x36, + 0x36, 0x1c, 0x1c, 0x08, 0x00, 0x00, 0x00, 0x00, + /* 87 */ 0x00, 0x00, 0x63, 0x63, 0x6b, 0x6b, 0x6b, 0x6b, + 0x7f, 0x36, 0x36, 0x36, 0x00, 0x00, 0x00, 0x00, + /* 88 */ 0x00, 0x00, 0x63, 0x63, 0x36, 0x36, 0x1c, 0x1c, + 0x36, 0x36, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 89 */ 0x00, 0x00, 0xc3, 0xc3, 0x66, 0x66, 0x3c, 0x3c, + 0x18, 0x18, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 90 */ 0x00, 0x00, 0x7f, 0x30, 0x30, 0x18, 0x18, 0x0c, + 0x0c, 0x06, 0x06, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 91 */ 0x00, 0x00, 0x3c, 0x0c, 0x0c, 0x0c, 0x0c, 0x0c, + 0x0c, 0x0c, 0x0c, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 92 */ 0x00, 0x00, 0x03, 0x02, 0x06, 0x04, 0x0c, 0x08, + 0x18, 0x10, 0x30, 0x20, 0x60, 0x00, 0x00, 0x00, + /* 93 */ 0x00, 0x00, 0x3c, 0x30, 0x30, 0x30, 0x30, 0x30, + 0x30, 0x30, 0x30, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 94 */ 0x00, 0x08, 0x1c, 0x36, 0x63, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 95 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0x00, 0x00, + /* 96 */ 0x00, 0x00, 0x0c, 0x0c, 0x18, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 97 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3e, 0x60, 0x7e, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 98 */ 0x00, 0x00, 0x03, 0x03, 0x03, 0x3b, 0x67, 0x63, + 0x63, 0x63, 0x67, 0x3b, 0x00, 0x00, 0x00, 0x00, + /* 99 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3e, 0x63, 0x03, + 0x03, 0x03, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 100 */ 0x00, 0x00, 0x60, 0x60, 0x60, 0x6e, 0x73, 0x63, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 101 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3e, 0x63, 0x63, + 0x7f, 0x03, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 102 */ 0x00, 0x00, 0x3c, 0x66, 0x06, 0x1f, 0x06, 0x06, + 0x06, 0x06, 0x06, 0x06, 0x00, 0x00, 0x00, 0x00, + /* 103 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x6e, 0x73, 0x63, + 0x63, 0x63, 0x73, 0x6e, 0x60, 0x63, 0x3e, 0x00, + /* 104 */ 0x00, 0x00, 0x03, 0x03, 0x03, 0x3b, 0x67, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 105 */ 0x00, 0x00, 0x0c, 0x0c, 0x00, 0x0c, 0x0c, 0x0c, + 0x0c, 0x0c, 0x0c, 0x38, 0x00, 0x00, 0x00, 0x00, + /* 106 */ 0x00, 0x00, 0x30, 0x30, 0x00, 0x30, 0x30, 0x30, + 0x30, 0x30, 0x30, 0x30, 0x30, 0x33, 0x1e, 0x00, + /* 107 */ 0x00, 0x00, 0x03, 0x03, 0x03, 0x63, 0x33, 0x1b, + 0x0f, 0x1f, 0x33, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 108 */ 0x00, 0x00, 0x0c, 0x0c, 0x0c, 0x0c, 0x0c, 0x0c, + 0x0c, 0x0c, 0x0c, 0x38, 0x00, 0x00, 0x00, 0x00, + /* 109 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x35, 0x6b, 0x6b, + 0x6b, 0x6b, 0x6b, 0x6b, 0x00, 0x00, 0x00, 0x00, + /* 110 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3b, 0x67, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 111 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3e, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 112 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3b, 0x67, 0x63, + 0x63, 0x63, 0x67, 0x3b, 0x03, 0x03, 0x03, 0x00, + /* 113 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x6e, 0x73, 0x63, + 0x63, 0x63, 0x73, 0x6e, 0x60, 0xe0, 0x60, 0x00, + /* 114 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3b, 0x67, 0x03, + 0x03, 0x03, 0x03, 0x03, 0x00, 0x00, 0x00, 0x00, + /* 115 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3e, 0x63, 0x0e, + 0x38, 0x60, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 116 */ 0x00, 0x00, 0x00, 0x0c, 0x0c, 0x3e, 0x0c, 0x0c, + 0x0c, 0x0c, 0x0c, 0x38, 0x00, 0x00, 0x00, 0x00, + /* 117 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 118 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x63, 0x63, 0x36, + 0x36, 0x1c, 0x1c, 0x08, 0x00, 0x00, 0x00, 0x00, + /* 119 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x63, 0x6b, 0x6b, + 0x6b, 0x3e, 0x36, 0x36, 0x00, 0x00, 0x00, 0x00, + /* 120 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x63, 0x36, 0x1c, + 0x1c, 0x1c, 0x36, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 121 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x63, 0x63, 0x36, + 0x36, 0x1c, 0x1c, 0x0c, 0x0c, 0x06, 0x03, 0x00, + /* 122 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x7f, 0x60, 0x30, + 0x18, 0x0c, 0x06, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 123 */ 0x00, 0x00, 0x70, 0x18, 0x18, 0x18, 0x18, 0x0e, + 0x18, 0x18, 0x18, 0x18, 0x70, 0x00, 0x00, 0x00, + /* 124 */ 0x00, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x18, 0x18, 0x18, 0x18, 0x00, 0x00, 0x00, + /* 125 */ 0x00, 0x00, 0x0e, 0x18, 0x18, 0x18, 0x18, 0x70, + 0x18, 0x18, 0x18, 0x18, 0x0e, 0x00, 0x00, 0x00, + /* 126 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x6e, 0x3b, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 127 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 128 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 129 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 130 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 131 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 132 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 133 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 134 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 135 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 136 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 137 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 138 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 139 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 140 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 141 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 142 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 143 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 144 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 145 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 146 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 147 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 148 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 149 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 150 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 151 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 152 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 153 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 154 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 155 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 156 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 157 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 158 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 159 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 160 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 161 */ 0x00, 0x00, 0x00, 0x00, 0x18, 0x18, 0x00, 0x18, + 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x00, 0x00, + /* 162 */ 0x00, 0x00, 0x00, 0x08, 0x08, 0x3e, 0x6b, 0x0b, + 0x0b, 0x0b, 0x6b, 0x3e, 0x08, 0x08, 0x00, 0x00, + /* 163 */ 0x00, 0x00, 0x1c, 0x36, 0x06, 0x06, 0x1f, 0x06, + 0x06, 0x07, 0x6f, 0x3b, 0x00, 0x00, 0x00, 0x00, + /* 164 */ 0x00, 0x00, 0x00, 0x00, 0x66, 0x3c, 0x66, 0x66, + 0x66, 0x3c, 0x66, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 165 */ 0x00, 0x00, 0xc3, 0xc3, 0x66, 0x66, 0x3c, 0x7e, + 0x18, 0x7e, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 166 */ 0x00, 0x00, 0x18, 0x18, 0x18, 0x18, 0x00, 0x00, + 0x18, 0x18, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 167 */ 0x00, 0x3c, 0x66, 0x0c, 0x1e, 0x33, 0x63, 0x66, + 0x3c, 0x18, 0x33, 0x1e, 0x00, 0x00, 0x00, 0x00, + /* 168 */ 0x00, 0x00, 0x36, 0x36, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 169 */ 0x00, 0x00, 0x3c, 0x42, 0x99, 0xa5, 0x85, 0xa5, + 0x99, 0x42, 0x3c, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 170 */ 0x00, 0x1e, 0x30, 0x3e, 0x33, 0x3b, 0x36, 0x00, + 0x3f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 171 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x6c, 0x36, 0x1b, + 0x1b, 0x36, 0x6c, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 172 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7f, 0x60, + 0x60, 0x60, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 173 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3c, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 174 */ 0x00, 0x00, 0x3c, 0x42, 0x9d, 0xa5, 0x9d, 0xa5, + 0xa5, 0x42, 0x3c, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 175 */ 0x00, 0x7e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 176 */ 0x00, 0x00, 0x1c, 0x36, 0x36, 0x1c, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 177 */ 0x00, 0x00, 0x00, 0x00, 0x18, 0x18, 0x7e, 0x18, + 0x18, 0x00, 0x7e, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 178 */ 0x00, 0x1e, 0x33, 0x18, 0x0c, 0x06, 0x3f, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 179 */ 0x00, 0x1e, 0x33, 0x18, 0x30, 0x33, 0x1e, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 180 */ 0x00, 0x30, 0x18, 0x0c, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 181 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x66, 0x66, 0x66, + 0x66, 0x66, 0x76, 0x6e, 0x06, 0x06, 0x03, 0x00, + /* 182 */ 0x00, 0x00, 0x7e, 0x2f, 0x2f, 0x2f, 0x2e, 0x28, + 0x28, 0x28, 0x28, 0x28, 0x00, 0x00, 0x00, 0x00, + /* 183 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, + 0x18, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 184 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x18, 0x30, 0x1e, 0x00, + /* 185 */ 0x00, 0x0c, 0x0e, 0x0c, 0x0c, 0x0c, 0x1e, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 186 */ 0x00, 0x1e, 0x33, 0x33, 0x33, 0x33, 0x1e, 0x00, + 0x3f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 187 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x1b, 0x36, 0x6c, + 0x6c, 0x36, 0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 188 */ 0x00, 0x10, 0x1c, 0x18, 0x18, 0x18, 0x00, 0x7f, + 0x00, 0x18, 0x1c, 0x1a, 0x3e, 0x18, 0x00, 0x00, + /* 189 */ 0x00, 0x10, 0x1c, 0x18, 0x18, 0x18, 0x00, 0x7f, + 0x00, 0x1c, 0x36, 0x18, 0x0c, 0x3e, 0x00, 0x00, + /* 190 */ 0x00, 0x1c, 0x36, 0x18, 0x36, 0x1c, 0x00, 0x7f, + 0x00, 0x18, 0x1c, 0x1a, 0x3e, 0x18, 0x00, 0x00, + /* 191 */ 0x00, 0x00, 0x00, 0x00, 0x0c, 0x0c, 0x00, 0x0c, + 0x0c, 0x06, 0x06, 0x03, 0x63, 0x3e, 0x00, 0x00, + /* 192 */ 0x0c, 0x18, 0x3e, 0x63, 0x63, 0x63, 0x7f, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 193 */ 0x18, 0x0c, 0x3e, 0x63, 0x63, 0x63, 0x7f, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 194 */ 0x08, 0x14, 0x3e, 0x63, 0x63, 0x63, 0x7f, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 195 */ 0x6e, 0x3b, 0x3e, 0x63, 0x63, 0x63, 0x7f, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 196 */ 0x36, 0x00, 0x3e, 0x63, 0x63, 0x63, 0x7f, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 197 */ 0x1c, 0x36, 0x3e, 0x63, 0x63, 0x63, 0x7f, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 198 */ 0x00, 0x00, 0xfe, 0x33, 0x33, 0x33, 0xff, 0x33, + 0x33, 0x33, 0x33, 0xf3, 0x00, 0x00, 0x00, 0x00, + /* 199 */ 0x00, 0x00, 0x3c, 0x66, 0x03, 0x03, 0x03, 0x03, + 0x03, 0x03, 0x66, 0x3c, 0x18, 0x30, 0x1e, 0x00, + /* 200 */ 0x0c, 0x18, 0x7f, 0x03, 0x03, 0x03, 0x3f, 0x03, + 0x03, 0x03, 0x03, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 201 */ 0x18, 0x0c, 0x7f, 0x03, 0x03, 0x03, 0x3f, 0x03, + 0x03, 0x03, 0x03, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 202 */ 0x08, 0x14, 0x7f, 0x03, 0x03, 0x03, 0x3f, 0x03, + 0x03, 0x03, 0x03, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 203 */ 0x36, 0x00, 0x7f, 0x03, 0x03, 0x03, 0x3f, 0x03, + 0x03, 0x03, 0x03, 0x7f, 0x00, 0x00, 0x00, 0x00, + /* 204 */ 0x0c, 0x18, 0x3c, 0x18, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x18, 0x18, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 205 */ 0x30, 0x18, 0x3c, 0x18, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x18, 0x18, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 206 */ 0x18, 0x24, 0x3c, 0x18, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x18, 0x18, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 207 */ 0x66, 0x00, 0x3c, 0x18, 0x18, 0x18, 0x18, 0x18, + 0x18, 0x18, 0x18, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 208 */ 0x00, 0x00, 0x1e, 0x36, 0x66, 0x66, 0x6f, 0x66, + 0x66, 0x66, 0x36, 0x1e, 0x00, 0x00, 0x00, 0x00, + /* 209 */ 0x6e, 0x3b, 0x63, 0x63, 0x67, 0x6f, 0x6f, 0x7b, + 0x7b, 0x73, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 210 */ 0x06, 0x0c, 0x3e, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 211 */ 0x30, 0x18, 0x3e, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 212 */ 0x08, 0x14, 0x3e, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 213 */ 0x6e, 0x3b, 0x3e, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 214 */ 0x36, 0x00, 0x3e, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 215 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x66, 0x3c, 0x18, + 0x3c, 0x66, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 216 */ 0x00, 0x20, 0x3e, 0x73, 0x73, 0x6b, 0x6b, 0x6b, + 0x6b, 0x67, 0x67, 0x3e, 0x02, 0x00, 0x00, 0x00, + /* 217 */ 0x0c, 0x18, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 218 */ 0x18, 0x0c, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 219 */ 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + /* 220 */ 0x36, 0x00, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 221 */ 0x30, 0x18, 0xc3, 0xc3, 0x66, 0x66, 0x3c, 0x3c, + 0x18, 0x18, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, + /* 222 */ 0x00, 0x00, 0x0f, 0x06, 0x3e, 0x66, 0x66, 0x66, + 0x66, 0x3e, 0x06, 0x0f, 0x00, 0x00, 0x00, 0x00, + /* 223 */ 0x00, 0x00, 0x1e, 0x33, 0x33, 0x1b, 0x33, 0x63, + 0x63, 0x63, 0x63, 0x3b, 0x00, 0x00, 0x00, 0x00, + /* 224 */ 0x00, 0x0c, 0x18, 0x30, 0x00, 0x3e, 0x60, 0x7e, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 225 */ 0x00, 0x30, 0x18, 0x0c, 0x00, 0x3e, 0x60, 0x7e, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 226 */ 0x00, 0x08, 0x1c, 0x36, 0x00, 0x3e, 0x60, 0x7e, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 227 */ 0x00, 0x00, 0x6e, 0x3b, 0x00, 0x3e, 0x60, 0x7e, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 228 */ 0x00, 0x00, 0x36, 0x36, 0x00, 0x3e, 0x60, 0x7e, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 229 */ 0x00, 0x1c, 0x36, 0x1c, 0x00, 0x3e, 0x60, 0x7e, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 230 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x6e, 0xdb, 0xd8, + 0xfe, 0x1b, 0xdb, 0x76, 0x00, 0x00, 0x00, 0x00, + /* 231 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x3e, 0x63, 0x03, + 0x03, 0x03, 0x63, 0x3e, 0x18, 0x30, 0x1e, 0x00, + /* 232 */ 0x00, 0x0c, 0x18, 0x30, 0x00, 0x3e, 0x63, 0x63, + 0x7f, 0x03, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 233 */ 0x00, 0x30, 0x18, 0x0c, 0x00, 0x3e, 0x63, 0x63, + 0x7f, 0x03, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 234 */ 0x00, 0x08, 0x1c, 0x36, 0x00, 0x3e, 0x63, 0x63, + 0x7f, 0x03, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 235 */ 0x00, 0x00, 0x36, 0x36, 0x00, 0x3e, 0x63, 0x63, + 0x7f, 0x03, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 236 */ 0x00, 0x06, 0x0c, 0x18, 0x00, 0x0c, 0x0c, 0x0c, + 0x0c, 0x0c, 0x0c, 0x38, 0x00, 0x00, 0x00, 0x00, + /* 237 */ 0x00, 0x18, 0x0c, 0x06, 0x00, 0x0c, 0x0c, 0x0c, + 0x0c, 0x0c, 0x0c, 0x38, 0x00, 0x00, 0x00, 0x00, + /* 238 */ 0x00, 0x08, 0x1c, 0x36, 0x00, 0x0c, 0x0c, 0x0c, + 0x0c, 0x0c, 0x0c, 0x38, 0x00, 0x00, 0x00, 0x00, + /* 239 */ 0x00, 0x00, 0x36, 0x36, 0x00, 0x0c, 0x0c, 0x0c, + 0x0c, 0x0c, 0x0c, 0x38, 0x00, 0x00, 0x00, 0x00, + /* 240 */ 0x00, 0x00, 0x2c, 0x18, 0x34, 0x60, 0x7c, 0x66, + 0x66, 0x66, 0x66, 0x3c, 0x00, 0x00, 0x00, 0x00, + /* 241 */ 0x00, 0x00, 0x6e, 0x3b, 0x00, 0x3b, 0x67, 0x63, + 0x63, 0x63, 0x63, 0x63, 0x00, 0x00, 0x00, 0x00, + /* 242 */ 0x00, 0x06, 0x0c, 0x18, 0x00, 0x3e, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 243 */ 0x00, 0x30, 0x18, 0x0c, 0x00, 0x3e, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 244 */ 0x00, 0x08, 0x1c, 0x36, 0x00, 0x3e, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 245 */ 0x00, 0x00, 0x6e, 0x3b, 0x00, 0x3e, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 246 */ 0x00, 0x00, 0x36, 0x36, 0x00, 0x3e, 0x63, 0x63, + 0x63, 0x63, 0x63, 0x3e, 0x00, 0x00, 0x00, 0x00, + /* 247 */ 0x00, 0x00, 0x00, 0x00, 0x18, 0x18, 0x00, 0x7e, + 0x00, 0x18, 0x18, 0x00, 0x00, 0x00, 0x00, 0x00, + /* 248 */ 0x00, 0x00, 0x00, 0x00, 0x20, 0x3e, 0x73, 0x6b, + 0x6b, 0x6b, 0x67, 0x3e, 0x02, 0x00, 0x00, 0x00, + /* 249 */ 0x00, 0x06, 0x0c, 0x18, 0x00, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 250 */ 0x00, 0x30, 0x18, 0x0c, 0x00, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 251 */ 0x00, 0x08, 0x1c, 0x36, 0x00, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 252 */ 0x00, 0x00, 0x36, 0x36, 0x00, 0x63, 0x63, 0x63, + 0x63, 0x63, 0x73, 0x6e, 0x00, 0x00, 0x00, 0x00, + /* 253 */ 0x00, 0x30, 0x18, 0x0c, 0x00, 0x63, 0x63, 0x36, + 0x36, 0x1c, 0x1c, 0x0c, 0x0c, 0x06, 0x03, 0x00, + /* 254 */ 0x00, 0x00, 0x0f, 0x06, 0x06, 0x3e, 0x66, 0x66, + 0x66, 0x66, 0x66, 0x3e, 0x06, 0x06, 0x0f, 0x00, + /* 255 */ 0x00, 0x00, 0x36, 0x36, 0x00, 0x63, 0x63, 0x36, + 0x36, 0x1c, 0x1c, 0x0c, 0x0c, 0x06, 0x03, 0x00 +}; + +Context2D* NewContext2D(I64 width, I64 height, + I64 bpp = 32) +{ // Create new Context2D. + switch (bpp) { + case 32: + break; + default: + return NULL; + break; + } + Context2D* ctx = CAlloc(sizeof(Context2D)); + ctx->width = width; + ctx->height = height; + ctx->bpp = bpp; + ctx->fb = CAlloc((width * height) * bpp / 8); + return ctx; +} + +U0 DelContext2D(Context2D* ctx) +{ + if (!ctx) { + return; + } + Free(ctx->fb); + Free(ctx); +} + +U0 Fill2D(Context2D* ctx, U32 color = 0) +{ // Fill a Context2D with color. + MemSetU32(ctx->fb, color, ctx->width * ctx->height); + return; +} + +U32 Peek2D(Context2D* ctx, I64 x, I64 y) +{ // Return RGBA value for pixel. + if (x < 0 || x > ctx->width - 1 || y < 0 || y > ctx->height - 1) + return Color(255, 255, 255); + return ctx->fb[(ctx->width * y) + x]; +} + +U0 Plot2D(Context2D* ctx, I64 x, I64 y, + U32 color) +{ // Plot a pixel with clipping. + if (x < 0 || x > ctx->width - 1 || y < 0 || y > ctx->height - 1) + return; + ctx->fb[(ctx->width * y) + x] = color; +} + +U0 VLine2D(Context2D* ctx, I64 x, I64 y, I64 y2, + U32 color) +{ // Draw a vertical line. + if (x > ctx->width || y > ctx->height) + return; + if (y2 < y) + return; + while (y < y2 + 1) { + Plot2D(ctx, x, y, color); + y++; + } +} + +U0 HLine2D(Context2D* ctx, I64 x, I64 y, I64 x2, + U32 color) +{ // Draw a horizontal line. + if (x2 < x) + return; + I64 width = x2 - x; + + MemSetU32(ctx->fb + (y * ctx->width) + x, color, + T(x + width > ctx->width, ctx->width - x, width)); +} + +U0 Line2D(Context2D* ctx, I64 x1, I64 y1, I64 x2, I64 y2, + U32 color) +{ // Draw an arbitrary line using Bresenham's algorithm. + x1 = Max(0, x1); + y1 = Max(0, y1); + x2 = Min(ctx->width, x2); + y2 = Min(ctx->height, y2); + if (x1 == x2) { + VLine2D(ctx, x1, y1, y2, color); + return; + } + if (y1 == y2) { + HLine2D(ctx, x1, y1, x2, color); + return; + } + I64 dx, sx, dy, sy; + I64 err, e2; + dx = Abs(x2 - x1); + sx = T(x1 < x2, 1, -1); + dy = -Abs(y2 - y1); + sy = T(y1 < y2, 1, -1); + err = dx + dy; + while (1) { + Plot2D(ctx, x1, y1, color); + if (x2 == x1 && y2 == y1) + break; + e2 = 2 * err; + if (e2 >= dy) { + err += dy; + x1 += sx; + } + if (e2 <= dx) { + err += dx; + y1 += sy; + } + } +} + +Context2D* MirroredHorz2D(Context2D* src) +{ + if (!src) + return NULL; + Context2D* dst = NewContext2D(src->width, src->height); + I64 x, y; + for (y = 0; y < src->height; y++) + for (x = src->width - 1; x > -1; x--) + Plot2D(dst, (src->width - 1) - x, y, Peek2D(src, x, y)); + return dst; +} + +Context2D* MirroredVert2D(Context2D* src) +{ + if (!src) + return NULL; + Context2D* dst = NewContext2D(src->width, src->height); + I64 x, y; + for (x = 0; x < src->width; x++) + for (y = src->height - 1; y > -1; y--) + Plot2D(dst, x, (src->height - 1) - y, Peek2D(src, x, y)); + return dst; +} + +I64 Blend2D(F64 alpha, F64 value1, F64 value2) +{ + return ToI64((1 - alpha) * value1 + alpha * value2); +} + +U0 Blot2D(Context2D* dst, I64 x, I64 y, Context2D* src) +{ + if (!src || !dst) + return; + I64 xx, yy; + I64 src_col, dst_col; + F64 alpha; + for (yy = 0; yy < src->height; yy++) { + for (xx = 0; xx < src->width; xx++) { + src_col = Peek2D(src, xx, yy); + dst_col = Peek2D(dst, x + xx, y + yy); + alpha = src_col.u8[3] / 128; // FIXME: Alpha blending not working correctly. + Plot2D(dst, x + xx, y + yy, Blend2D(alpha, dst_col, src_col)); + } + } +} + +U0 @patch_blend_rect_2d(U64 addr_start, I64 size, U64 jmp_offset, + U64 table_offset) +{ + U8* patch_ptr = addr_start; + I64 i; + for (i = 0; (addr_start + table_offset - i) % 16; i++) { + patch_ptr[jmp_offset]--; + } + if (i) + MemCpy(addr_start + table_offset - i, addr_start + table_offset, + size - table_offset); +} + +U0 BlendRect2D(Context2D* rect, Context2D* ctx) +{ + U64 reg R10 d = ctx->fb; + U64 reg R11 s1 = rect->fb; + U64 reg R14 s2 = ctx->fb; + I64 reg RSI w = ctx->width; + I64 reg RDI h = ctx->height; + no_warn d, s1, s2, w, h; + asm { + MOV RAX, R10 + MOV RDX, R11 + MOV RCX, R14 + DU8 + 0x0f, 0xaf, 0xf7, 0xc1, 0xfe, 0x02, 0xeb, 0x7f, + 0x66, 0x66, 0x2e, 0x0f, 0x1f, 0x84, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x0f, 0x1f, 0x40, 0x00, 0x90, + 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, + 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x03, + 0x80, 0x03, 0x80, 0x07, 0x80, 0x07, 0x80, 0x0b, + 0x80, 0x0b, 0x80, 0x0f, 0x80, 0x0f, 0x80, 0x00, + 0x80, 0x02, 0x80, 0x04, 0x80, 0x06, 0x80, 0x08, + 0x80, 0x0a, 0x80, 0x0c, 0x80, 0x0e, 0x80, 0x01, + 0x80, 0x03, 0x80, 0x05, 0x80, 0x07, 0x80, 0x09, + 0x80, 0x0b, 0x80, 0x0d, 0x80, 0x0f, 0x80, 0xff, + 0x00, 0xff, 0x00, 0xff, 0x00, 0xff, 0x00, 0xff, + 0x00, 0xff, 0x00, 0xff, 0x00, 0xff, 0x00, 0x01, + 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, + 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x00, + 0x00, 0x00, 0xff, 0x00, 0x00, 0x00, 0xff, 0x00, + 0x00, 0x00, 0xff, 0x00, 0x00, 0x00, 0xff, 0x0f, + 0x10, 0x0a, 0x0f, 0x10, 0x11, 0x66, 0x0f, 0x6f, + 0xd9, 0x66, 0x0f, 0x38, 0x00, 0x1d, 0x8d, 0xff, + 0xff, 0xff, 0x66, 0x0f, 0x6f, 0xe3, 0x66, 0x0f, + 0xef, 0x25, 0xb1, 0xff, 0xff, 0xff, 0x66, 0x0f, + 0x6f, 0xe9, 0x66, 0x0f, 0x6f, 0xf2, 0x66, 0x0f, + 0x38, 0x00, 0x0d, 0x80, 0xff, 0xff, 0xff, 0x66, + 0x0f, 0xd5, 0xcb, 0x66, 0x0f, 0x38, 0x00, 0x15, + 0x73, 0xff, 0xff, 0xff, 0x66, 0x0f, 0xd5, 0xd4, + 0x66, 0x0f, 0x38, 0x00, 0x2d, 0x76, 0xff, 0xff, + 0xff, 0x66, 0x0f, 0xd5, 0xeb, 0x66, 0x0f, 0xfd, + 0xd1, 0x66, 0x0f, 0xe4, 0x15, 0x86, 0xff, 0xff, + 0xff, 0x66, 0x0f, 0x38, 0x00, 0x35, 0x5d, 0xff, + 0xff, 0xff, 0x66, 0x0f, 0xd5, 0xf4, 0x66, 0x0f, + 0xfd, 0xf5, 0x66, 0x0f, 0xe4, 0x35, 0x6d, 0xff, + 0xff, 0xff, 0x66, 0x0f, 0x71, 0xf6, 0x08, 0x66, + 0x0f, 0xeb, 0xd6, 0x66, 0x0f, 0xeb, 0x15, 0x6c, + 0xff, 0xff, 0xff, 0x0f, 0x11, 0x10, 0x48, 0x83, + 0xc2, 0x10, 0x48, 0x83, 0xc1, 0x10, 0x48, 0x83, + 0xc0, 0x10, 0xff, 0xce, 0x0f, 0x85, 0x65, 0xff, + 0xff, 0xff, 0x90; + } +} + +@patch_blend_rect_2d(&BlendRect2D, sizeof(BlendRect2D), 0x3D, 0x5D); + +U0 CopyRect2D(Context2D* ctx, I64 x, I64 y, + Context2D* rect) +{ // Copy rect with clipping. + if (x > ctx->width - 1 || y > ctx->height - 1) + return; + U8* ctx_pos = ctx->fb; + U8* rect_pos = rect->fb; + I64 rect_row = 0; + I64 rect_y_ofs = 0; + I64 rect_x_ofs = 0; + I64 clip_y = 0; + U8* rect_line; + I64 bpp = 32 / 8; + + // Handle horizontal clipping left + while (x < 0) { + rect_x_ofs++; + x++; + } + + // Handle vertical clipping top + while (y < 0) { + rect_pos += (rect->width) * bpp; + rect_y_ofs++; + y++; + } + + // default, clip line to copy as width-left off screen + rect_line = rect->width - rect_x_ofs; + + if (-rect_x_ofs + x + rect->width >= ctx->width) { + rect_line -= ((-rect_x_ofs + x + rect->width) - ctx->width); + } + + rect_pos += (rect_x_ofs)*bpp; + clip_y = y; + while (rect_row < (rect->height - rect_y_ofs) && clip_y < ctx->height) { + MemCpyU32(ctx_pos + (y * ((ctx->width) * bpp)) + (x * bpp), rect_pos, + (rect_line)); + ctx_pos += (ctx->width) * bpp; + rect_pos += (rect->width) * bpp; + clip_y++; + rect_row++; + } +} + +U0 @graphics2d_get_gradient_steps(U32 start, U32 end, U32* gradient) +{ + I64 a2 = start.u8[3]; + I64 r2 = start.u8[2]; + I64 g2 = start.u8[1]; + I64 b2 = start.u8[0]; + I64 a1 = end.u8[3]; + I64 r1 = end.u8[2]; + I64 g1 = end.u8[1]; + I64 b1 = end.u8[0]; + I64 i; + F64 c = 1.0; + for (i = 0; i < 256; i++) { + gradient[i] = Color(r1 / 255.0 * c + r2 / 255.0 * (255 - c), + g1 / 255.0 * c + g2 / 255.0 * (255 - c), + b1 / 255.0 * c + b2 / 255.0 * (255 - c), + a1 / 255.0 * c + a2 / 255.0 * (255 - c)); + c += 1.0; + } +} + +U0 HGradientRect2D(Context2D* ctx, I64 x, I64 y, I64 w, I64 h, U32 from, + U32 to) +{ + U32 gradient[256]; + F64 gradient_index; + I64 xx; + @graphics2d_get_gradient_steps(from, to, &gradient); + for (xx = 0; xx < w; xx++) { + gradient_index = ToF64(256.0 / ToF64(w)) * ToF64(xx); + Line2D(ctx, x + xx, y, x + xx, y + h - 1, gradient[ToI64(gradient_index)]); + } +} + +U0 Rect2D(Context2D* ctx, I64 x, I64 y, I64 w, I64 h, + U32 color) +{ // Draw a rectangle fill. + Context2D* tmpctx = NewContext2D(Max(4, w), Max(4, h)); + Fill2D(tmpctx, color); + CopyRect2D(ctx, x, y, tmpctx); + DelContext2D(tmpctx); +} + +U0 ConsolePrint2D(Context2D* ctx, I64 x, I64 y, + U32 color = Color(255, 255, 255), U32 colorbg = 0, U8* fmt, + ...) +{ // Print formatted string using console font. + Bool skip; + U8* buf = StrPrintJoin(NULL, fmt, argc, argv); + U8* str = buf; + I64 orig_x = x; + I64 xx, yy; + U64* chr = console_font; + while (*str) { + skip = FALSE; + if (*str == '\n') { + skip = TRUE; + y += 16; + x = orig_x - 8; + } + for (yy = 0; yy < 16; yy++) { + for (xx = 0; xx < 8; xx++) { + if (chr[(*str) * 2].u8[yy] & 1 << xx == 1 << xx && !skip) + Plot2D(ctx, x + xx, y + yy, color); + else + Plot2D(ctx, x + xx, y + yy, colorbg); + } + } + x += 8; + str++; + } + Free(buf); +} + +U0 PutChar2D(Context2D* ctx, BitmapFont* font, I64 x, I64 y, + U32 color = Color(0, 0, 0), I64 char) +{ + if (!ctx) + return; + I64 xx; + I64 yy; + I64 char_index = 0; + while (char != font->char_map[char_index]) + char_index++; + + for (yy = 0; yy < 16; yy++) { + for (xx = 0; xx < 16; xx++) { + if (font->bitmap[(char_index * 16) + yy] & 0x8000 >> xx == 0x8000 >> xx && char != ' ') { + Plot2D(ctx, x + xx, y + yy, color); + } + } + } +} + +I64 PutS2D(Context2D* ctx, BitmapFont* font, I64 x, I64 y, + U32 color = Color(0, 0, 0), I64 max_width = -1, U8* buf) +{ + U8* str = buf; + + I64 origin_x; + I64 origin_y; + I64 xx, yy; + Bool char_space; + + I64 char_index; + I64 insert_space; + I64 pass; + + origin_x = x; + origin_y = y; + if (max_width > -1) + max_width = Min(ctx->width, max_width); + + for (pass = 0; pass < 2; pass++) { + x = origin_x; + y = origin_y; + str = buf; + while (*str) { + char_space = FALSE; + switch (*str) { + case '\n': + x = origin_x; + if (font->line_height) + y += Min(16, font->line_height); + else + y += 16; + goto @print2d_next_char; + break; + case ' ': + char_space = TRUE; + break; + default: + break; + } + char_index = 0; + while (*str != font->char_map[char_index]) + char_index++; + insert_space = 0; + for (yy = 0; yy < 16; yy++) { + for (xx = 0; xx < 16; xx++) { + if (font->bitmap[(char_index * 16) + yy] & 0x8000 >> xx == 0x8000 >> xx && !char_space) { + insert_space = Max(xx, insert_space); + if (pass && ctx) + Plot2D(ctx, x + xx, y + yy, color); + } + } + } + if (char_space) + insert_space = 4; + @print2d_next_char : str++; + if (str > buf && str[-1] != '\n') { + if (*str) { + x += insert_space + 2; + } else { + x += insert_space; + } + } + if (max_width == -1) { + } else if (x > max_width) { + StrCpy(&str[-4], "..."); + break; + } + } + } + return x; +} + +I64 Print2D(Context2D* ctx, BitmapFont* font, I64 x, I64 y, + U32 color = Color(0, 0, 0), I64 max_width = -1, U8* fmt, + ...) +{ // Print formatted string using BitmapFont. + U8* buf; + if (argc) + buf = StrPrintJoin(NULL, fmt, argc, argv); + else + buf = StrNew(fmt); + I64 retval = PutS2D(ctx, font, x, y, color, max_width, buf); + Free(buf); + return retval; +} + +Context2D* FastBoxBlur2D(Context2D* img, I64 radius) +{ + /* + Algorithm by Wojciech Jarosz, Implementation by Ferris Ateniese + http://elynxsdk.free.fr/ext-docs/Blur/Fast_box_blur.pdf + https://gist.github.com/LionRoar/12d625bee5882abb339dc7102ad6fe32#file-fastboxblur-cs + */ + I64 kSize = radius; + I64 c, i, j, x, y; + I64 bpp = 32; + no_warn bpp; + U32 tmpColor, tmp_nColor, tmp_pColor, plot_color; + F64 hSum[4]; + F64 tSum[4]; + F64 iAvg[4]; + + if (kSize % 2 == 0) + kSize++; + + Context2D* Hblur = NewContext2D(img->width, img->height); + MemCpyU32(Hblur->fb, img->fb, img->width * img->height); + + F64 Avg = 1.0 / kSize; + + for (j = 0; j < img->height; j++) { + for (c = 0; c < 4; c++) { + hSum[c] = 0.0; + iAvg[c] = 0.0; + } + for (x = 0; x < kSize; x++) { + tmpColor = Peek2D(img, x, j); + hSum[3] += tmpColor.u8[3]; + hSum[2] += tmpColor.u8[2]; + hSum[1] += tmpColor.u8[1]; + hSum[0] += tmpColor.u8[0]; + } + iAvg[3] = hSum[3] * Avg; + iAvg[2] = hSum[2] * Avg; + iAvg[1] = hSum[1] * Avg; + iAvg[0] = hSum[0] * Avg; + for (i = 0; i < img->width; i++) { + if (i - kSize / 2 >= 0 && i + 1 + kSize / 2 < img->width) { + tmp_pColor = Peek2D(img, i - kSize / 2, j); + hSum[3] -= tmp_pColor.u8[3]; + hSum[2] -= tmp_pColor.u8[2]; + hSum[1] -= tmp_pColor.u8[1]; + hSum[0] -= tmp_pColor.u8[0]; + tmp_nColor = Peek2D(img, i + 1 + kSize / 2, j); + hSum[3] += tmp_nColor.u8[3]; + hSum[2] += tmp_nColor.u8[2]; + hSum[1] += tmp_nColor.u8[1]; + hSum[0] += tmp_nColor.u8[0]; + iAvg[3] = hSum[3] * Avg; + iAvg[2] = hSum[2] * Avg; + iAvg[1] = hSum[1] * Avg; + iAvg[0] = hSum[0] * Avg; + } + plot_color.u8[3] = ToI64(iAvg[3]); + plot_color.u8[2] = ToI64(iAvg[2]); + plot_color.u8[1] = ToI64(iAvg[1]); + plot_color.u8[0] = ToI64(iAvg[0]); + Plot2D(Hblur, i, j, plot_color); + } + } + + Context2D* total = NewContext2D(Hblur->width, Hblur->height); + MemCpyU32(total->fb, Hblur->fb, Hblur->width * Hblur->height); + + for (i = 0; i < Hblur->width; i++) { + for (c = 0; c < 4; c++) { + tSum[c] = 0.0; + iAvg[c] = 0.0; + } + for (y = 0; y < kSize; y++) { + tmpColor = Peek2D(Hblur, i, y); + tSum[3] += tmpColor.u8[3]; + tSum[2] += tmpColor.u8[2]; + tSum[1] += tmpColor.u8[1]; + tSum[0] += tmpColor.u8[0]; + } + iAvg[3] = tSum[3] * Avg; + iAvg[2] = tSum[2] * Avg; + iAvg[1] = tSum[1] * Avg; + iAvg[0] = tSum[0] * Avg; + for (j = 0; j < Hblur->height; j++) { + if (j - kSize / 2 >= 0 && j + 1 + kSize / 2 < Hblur->height) { + tmp_pColor = Peek2D(Hblur, i, j - kSize / 2); + tSum[3] -= tmp_pColor.u8[3]; + tSum[2] -= tmp_pColor.u8[2]; + tSum[1] -= tmp_pColor.u8[1]; + tSum[0] -= tmp_pColor.u8[0]; + tmp_nColor = Peek2D(Hblur, i, j + 1 + kSize / 2); + + tSum[3] += tmp_nColor.u8[3]; + tSum[2] += tmp_nColor.u8[2]; + tSum[1] += tmp_nColor.u8[1]; + tSum[0] += tmp_nColor.u8[0]; + iAvg[3] = tSum[3] * Avg; + iAvg[2] = tSum[2] * Avg; + iAvg[1] = tSum[1] * Avg; + iAvg[0] = tSum[0] * Avg; + } + plot_color.u8[3] = ToI64(iAvg[3]); + plot_color.u8[2] = ToI64(iAvg[2]); + plot_color.u8[1] = ToI64(iAvg[1]); + plot_color.u8[0] = ToI64(iAvg[0]); + Plot2D(total, i, j, plot_color); + } + } + DelContext2D(Hblur); + return total; +} + +U8 @scale2d_get_byte(I64 value, I64 n) { return (value >> (n * 8) & 0xFF); } + +U32 @scale2d_get_pixel(Context2D* img, I64 x, I64 y) +{ + return img->fb[(y * img->width) + x]; +} + +F64 @scale2d_lerp(F64 s, F64 e, F64 t) { return s + (e - s) * t; } + +F64 @scale2d_blerp(F64 c00, F64 c10, F64 c01, F64 c11, F64 tx, F64 ty) +{ + return @scale2d_lerp(@scale2d_lerp(c00, c10, tx), @scale2d_lerp(c01, c11, tx), + ty); +} + +U0 @scale2d_put_pixel(Context2D* img, I64 x, I64 y, U32 color) +{ + img->fb[(y * img->width) + x] = color; +} + +I64 X1Pos2D(Context2D* src) +{ + I64 x = 0; + I64 y = 0; + U32 color; + while (x < src->width) { + y = 0; + while (y < src->height) { + color = Peek2D(src, x, y); + if (color.u8[3]) + return x; + y++; + } + x++; + } + return -1; +} + +I64 Y1Pos2D(Context2D* src) +{ + I64 x = 0; + I64 y = 0; + U32 color; + while (y < src->height) { + x = 0; + while (x < src->width) { + color = Peek2D(src, x, y); + if (color.u8[3]) + return y; + x++; + } + y++; + } + return -1; +} + +I64 X2Pos2D(Context2D* src) +{ + I64 x = src->width - 1; + I64 y = src->height - 1; + U32 color; + while (x > -1) { + y = src->height - 1; + while (y > -1) { + color = Peek2D(src, x, y); + if (color.u8[3]) + return x; + y--; + } + x--; + } + return -1; +} + +I64 Y2Pos2D(Context2D* src) +{ + I64 x = src->width - 1; + I64 y = src->height - 1; + U32 color; + while (y > -1) { + x = src->width - 1; + while (x > -1) { + color = Peek2D(src, x, y); + if (color.u8[3]) + return y; + x--; + } + y--; + } + return -1; +} + +Context2D* ClipToRect2D(Context2D* src) +{ + I64 x1, y1, x2, y2; + x1 = X1Pos2D(src); + y1 = Y1Pos2D(src); + x2 = X2Pos2D(src); + y2 = Y2Pos2D(src); + Context2D* dst = NewContext2D(x2 - x1, y2 - y1); + CopyRect2D(dst, -x1, -y1, src); + return dst; +} + +Context2D* Scale2D(Context2D* src, F64 scale_x, F64 scale_y) +{ + I64 newWidth = ToI64(src->width * scale_x); + I64 newHeight = ToI64(src->height * scale_y); + Context2D* dst = NewContext2D(newWidth, newHeight); + I64 x, y; + F64 fw, fh; + F64 gx; + F64 gy; + I64 gxi; + I64 gyi; + U32 result; + U32 c00; + U32 c10; + U32 c01; + U32 c11; + U64 i; + fw = ToF64(newWidth); + fh = ToF64(newHeight); + for (x = 0, y = 0; y < newHeight; x++) { + if (x > newWidth) { + x = 0; + y++; + } + gx = x / fw * (src->width - 1); + gy = y / fh * (src->height - 1); + gxi = ToI64(gx); + gyi = ToI64(gy); + result = 0; + c00 = @scale2d_get_pixel(src, gxi, gyi); + c10 = @scale2d_get_pixel(src, gxi + 1, gyi); + c01 = @scale2d_get_pixel(src, gxi, gyi + 1); + c11 = @scale2d_get_pixel(src, gxi + 1, gyi + 1); + for (i = 0; i < 4; i++) { + result.u8[i] = ToI64( + @scale2d_blerp(@scale2d_get_byte(c00, i), @scale2d_get_byte(c10, i), + @scale2d_get_byte(c01, i), @scale2d_get_byte(c11, i), + gx - gxi, gy - gyi) + << (8 * i)); + } + @scale2d_put_pixel(dst, x, y, result); + } + return dst; +} + +Context2D* Rotate2D(Context2D* src, F64 angle) +{ // FIXME: This is just awful... + Context2D* dst = NewContext2D(src->width * 4, src->height * 4); + Fill2D(dst, 0); + CDC* dc[4]; + CSprite* s[4]; + I64 i, x, y; + U32 color; + for (i = 0; i < 4; i++) { + dc[i] = DCNew(src->width * 4, src->height * 4); + } + for (y = 0; y < src->height; y++) { + for (x = 0; x < src->width; x++) { + color = Peek2D(src, x, y); + for (i = 0; i < 4; i++) { + dc[i]->color = Min(color.u8[i], 254); + GrPlot(dc[i], x, y); + } + } + } + for (i = 0; i < 4; i++) { + s[i] = DC2Sprite(dc[i]); + DCFill(dc[i], 0); + Sprite3ZB(dc[i], src->width * 2, src->height * 2, 0, s[i], + angle * 0.017499999999); + } + for (y = 0; y < src->height * 4; y++) { + for (x = 0; x < src->width * 4; x++) { + for (i = 0; i < 4; i++) { + color.u8[i] = GrPeek(dc[i], x, y); + } + Plot2D(dst, x, y, color); + } + } + for (i = 0; i < 4; i++) { + DCDel(dc[i]); + Free(s[i]); + } + Context2D* dst2 = ClipToRect2D(dst); + DelContext2D(dst); + return dst2; +} + +class @graphics2d +{ + Context2D* fb; + + Context2D* (*FrameBufferContext2D)(); + U0(*Init) + (); + U0(*Flip) + (Context2D * ctx); +}; + +@graphics2d Graphics2D; + +U0 @graphics2d_init() +{ + Graphics2D.fb = CAlloc(sizeof(Context2D)); + Graphics2D.fb->width = Display.width; + Graphics2D.fb->height = Display.height; + Graphics2D.fb->bpp = Display.bpp; + Graphics2D.fb->fb = Display.fb; + Fill2D(Graphics2D.fb, 0x0); +} + +U0 @graphics2d_flip(Context2D* ctx) +{ + MemCpyU32(Graphics2D.fb->fb, ctx->fb, Display.width * Display.height); +} + +Context2D @graphics2d_get_framebuffer_context2d() { return Graphics2D.fb; } + +Graphics2D.FrameBufferContext2D = &@graphics2d_get_framebuffer_context2d; +Graphics2D.Init = &@graphics2d_init; +Graphics2D.Flip = &@graphics2d_flip; + +"graphics2d "; \ No newline at end of file diff --git a/System/Libraries/Gui.HC b/System/Libraries/Gui.HC new file mode 100644 index 0000000..1c2f34c --- /dev/null +++ b/System/Libraries/Gui.HC @@ -0,0 +1,398 @@ +extern class Widget; +extern class Window; + +#define WIN_FLAGS_NULL 0x0 +#define WIN_FLAGS_NO_REINDEX 0x1 // Wallpaper, taskbar, etc. +#define WIN_FLAGS_RESIZABLE 0x2 +#define WIN_FLAGS_MOVABLE 0x4 +#define WIN_FLAGS_ICON 0x8 +#define WIN_FLAGS_TITLE_BAR 0x10 +#define WIN_FLAGS_MIN_BUTTON 0x20 +#define WIN_FLAGS_MAX_BUTTON 0x40 +#define WIN_FLAGS_CLOSE_BUTTON 0x80 + +#define WIN_FLAGS_MINIMIZED 0x100 +#define WIN_FLAGS_MAXIMIZED 0x200 +#define WIN_FLAGS_HIDDEN 0x400 +#define WIN_FLAGS_NOHILIGHT 0x800 + +#define WIN_FLAGS_SKIP 0x1000 +#define WIN_FLAGS_NOFILL 0x2000 +#define WIN_FLAGS_MENU 0x4000 + +#define WIN_FLAGS_MAX 0x10000 + +#define WIN_SIGNATURE 0x1596e3c1c62c34b929d75cded8c0 + +#define WIN_FLAGS_DEFAULT \ + (WIN_FLAGS_RESIZABLE | WIN_FLAGS_MOVABLE | WIN_FLAGS_ICON | WIN_FLAGS_TITLE_BAR | WIN_FLAGS_MIN_BUTTON | WIN_FLAGS_MAX_BUTTON | WIN_FLAGS_CLOSE_BUTTON) + +class @widget_callbacks +{ + U0 (*change)(Widget* widget); + U0 (*clicked)(Widget* widget); + U0 (*repaint)(Widget* widget); +}; + +class @widget_origin +{ + I64 x; + I64 y; + I64 width; + I64 height; + I64 mouse_x; + I64 mouse_y; +}; + +class Widget { + Bool change; + I64 id; + I64 type; + I64 x; + I64 y; + I64 width; + I64 height; + I64 opacity; + U64 flags; + U8* tag; + Widget* echo; + Window* parent_win; + Context2D* backing_store; + Context2D* pointer; + @widget_callbacks callback; + @widget_origin origin; +}; + +class @window_widgets_list +{ + @window_widgets_list* prev; + @window_widgets_list* next; + Widget* widget; +}; + +class @window_origin +{ + I64 x; + I64 y; + I64 width; + I64 height; + I64 mouse_x; + I64 mouse_y; +}; + +class @window_position +{ + I64 x; + I64 y; +} + +class @window_buttons +{ + Bool minimize; + Bool maximize; + Bool close; +}; + +class @window_callbacks +{ + U0 (*minimize)(Window* win); + U0 (*maximize)(Window* win); + U0 (*mouseat)(Window* win); + U0 (*keypress)(Window* win, I64 key); + U0 (*repaint)(Window* win); + U0 (*close)(Window* win); +}; + +class @window_mouse +{ + I64 x; + I64 y; + Bool left; + Bool right; +}; + +class @window_event +{ // FIXME: Better name? + I64 x; + I64 y; + Bool left; + Bool right; +}; + +class Window { + U64 signature; + Bool alpha; + Bool refresh; + Bool repainting; + CTask* client; + I64 x; + I64 y; + I64 width; + I64 height; + I64 opacity; + U64 flags; + U8 title[512]; + I64 title_bar_x; + I64 title_bar_width; + I64 min_width; + I64 min_height; + Context2D* icon; + Context2D* backing_store; + Context2D* pointer; + Context2D* render_ctx; + Context2D* resize_ctx; + Widget* mouse_down_widget; + Widget* focused_widget; + Widget* hovered_widget; + @window_buttons button; + @window_callbacks callback; + @window_origin origin; + @window_mouse mouse; + @window_event left_btn_down; // FIXME: put these in a Window.event.xxx class? + @window_event left_btn_up; + @window_event right_btn_down; + @window_event right_btn_up; + @window_widgets_list* widget; +}; + +class @gui_widget +{ + Bool (*IsHovered)(Window* win, Widget* widget); + U0 (*SetCallback)(Widget* widget, U8* name, U64 callback); + U0 (*SetEcho)(Widget* widget, Widget* echo); + U0 (*SetFont)(Widget* widget, U8* font_name); + U0 (*SetMousePointer)(Widget* widget, Context2D* pointer); + U0 (*ClearMousePointer)(Widget* widget); + U0 (*SetOpacity)(Widget* widget, I64 opacity); + U0 (*SetText)(Widget* widget, U8* text); +}; + +class @gui_window +{ + U0 (*Center)(Window* win, Bool horz = TRUE, Bool vert = TRUE); + U0 (*DisableAlphaChannel)(Window* win); + U0 (*EnableAlphaChannel)(Window* win); + U0 (*Hide)(Window* win); + Bool (*IsHovered)(Window* win); + Bool (*IsVisible)(Window* win); + U0 (*SetCallback)(Window* win, U8* name, U64 callback); + U0 (*SetFocus)(Window* win); + U0 (*SetIcon)(Window* win, Context2D* icon); + U0 (*SetMousePointer)(Window* win, Context2D* pointer); + U0 (*ClearMousePointer)(Window* win); + U0 (*SetOpacity)(Window* win, I64 opacity); + U0 (*SetTitle)(Window* win, U8* text); + U0 (*SetPosition)(Window* win, I64 x, I64 y); + U0 (*SetZIndex)(Window* win, I64 index); + U0 (*Show)(Window* win); + U0 (*Refresh)(Window* win); +}; + +class @gui +{ + @gui_widget Widget; + @gui_window Window; + U0 (*App)(); + Widget* (*InitWidget)(Widget* widget, Window* win, I64 type, I64 x, I64 y, + I64 width, I64 height); + Widget* (*CreateWidget)(Window* win, I64 type, I64 x, I64 y, I64 width, + I64 height); +}; + +@gui Gui; + +I64 @gui_app_header_size; +U8* @gui_app_header_data = FileRead("M:/Include/Gui.HC", &@gui_app_header_size); + +U0 @gui_app() +{ + CDoc* @gui_app_header_doc = DocNew; + DocLoad(@gui_app_header_doc, @gui_app_header_data, @gui_app_header_size); + ExeDoc(@gui_app_header_doc); +} + +Bool @gui_window_flag_is_set(Window* win, U64 flag) +{ + if (!win) + return FALSE; + if (win->flags & flag == flag) + return TRUE; + return FALSE; +} + +Bool @gui_window_is_hovered(Window* win) +{ + if (Mouse.x > win->x && Mouse.x < win->x + win->width && Mouse.y > win->y && Mouse.y < win->y + win->height) + return TRUE; + return FALSE; +} + +Bool @gui_window_is_visible(Window* win) +{ + if (!win) + return NULL; + return !@gui_window_flag_is_set(win, WIN_FLAGS_HIDDEN); +} + +U0 @gui_widget_destroy(Widget* widget) +{ + Window* win = widget->parent_win; + @window_widgets_list* widgets = win->widget; + @window_widgets_list* prev; + @window_widgets_list* next; + while (widgets) { + if (widgets->widget == widget) { + prev = widgets->prev; + next = widgets->next; + if (prev) + prev->next = next; + if (next) + next->prev = prev; + // FIXME: Free widget and child data + widget->type = NULL; + } + widgets = widgets->next; + } +} + +U0 @gui_widget_repaint(Window* win, Widget* widget, I64 type) { } + +U0 @gui_window_repaint(Window* win, I64 type) +{ + @system_log(Fs, "Repainting window 0x%08x [%s]", win, win->title); +} + +U0 @gui_window_center(Window* win, Bool horz = TRUE, Bool vert = TRUE) +{ + if (!win) + return; + I64 x = win->x; + I64 y = win->y; + if (horz) + x = (Display.Width() / 2) - (win->width / 2); + if (vert) + y = (Display.Height() / 2) - (win->height / 2); + win->x = x; + win->y = y; +} + +U0 @gui_window_set_position(Window* win, I64 x, I64 y) +{ + if (!win) + return; + win->x = x; + win->y = y; +} + +U0 @gui_window_set_icon(Window* win, Context2D* icon) +{ + if (!win || !icon) + return; + Bool refresh = FALSE; + if (win->icon != icon) + refresh = TRUE; + win->icon = icon; + if (refresh) + Gui.Window.Refresh(win); +} + +U0 @gui_window_set_mouse_pointer(Window* win, Context2D* pointer) +{ + if (!win) + return; + win->pointer = pointer; +} + +U0 @gui_window_clear_mouse_pointer(Window* win) +{ + if (!win) + return; + win->pointer = NULL; +} + +U0 @gui_window_set_opacity(Window* win, I64 opacity) +{ + if (!win) + return; + Bool refresh = FALSE; + if (win->opacity != opacity) + refresh = TRUE; + win->opacity = ClampI64(opacity, 0, 255); + if (refresh) + Gui.Window.Refresh(win); +} + +U0 @gui_window_set_title(Window* win, U8* text) +{ + if (!win || !text) + return; + if (!StrLen(text)) + return; + if (StrLen(text) > 511) { + MemCpy(&win->title, text, 512); + win->title[511] = NULL; + return; + } + StrCpy(&win->title, text); + Gui.Window.Refresh(win); +} + +U0 @gui_window_callback_close(Window* win) { } + +U0 @gui_window_callback_maximize(Window* win) +{ + win->flags |= WIN_FLAGS_MAXIMIZED; +} + +U0 @gui_window_callback_minimize(Window* win) +{ + win->flags |= WIN_FLAGS_MINIMIZED; +} + +U0 @gui_window_disable_alpha_channel(Window* win) { win->alpha = FALSE; } + +U0 @gui_window_enable_alpha_channel(Window* win) { win->alpha = TRUE; } + +U0 @gui_window_hide(Window* win) { win->flags |= WIN_FLAGS_HIDDEN; } + +U0 @gui_window_show(Window* win) +{ + win->flags &= ~WIN_FLAGS_HIDDEN; + win->flags &= ~WIN_FLAGS_MINIMIZED; +} + +U0 @gui_window_set_callback(Window* win, U8* name, U64 callback) +{ + if (!win || !name || !callback) + return; + if (!StrCmp(name, "minimize")) + win->callback.minimize = callback; + if (!StrCmp(name, "maximize")) + win->callback.maximize = callback; + if (!StrCmp(name, "mouseat")) + win->callback.mouseat = callback; + if (!StrCmp(name, "keypress")) + win->callback.keypress = callback; + if (!StrCmp(name, "repaint")) + win->callback.repaint = callback; + if (!StrCmp(name, "close")) + win->callback.close = callback; +} + +Gui.App = &@gui_app; +Gui.Window.Center = &@gui_window_center; +Gui.Window.DisableAlphaChannel = &@gui_window_disable_alpha_channel; +Gui.Window.EnableAlphaChannel = &@gui_window_enable_alpha_channel; +Gui.Window.Hide = &@gui_window_hide; +Gui.Window.IsHovered = &@gui_window_is_hovered; +Gui.Window.IsVisible = &@gui_window_is_visible; +Gui.Window.SetCallback = &@gui_window_set_callback; +Gui.Window.SetIcon = &@gui_window_set_icon; +Gui.Window.SetMousePointer = &@gui_window_set_mouse_pointer; +Gui.Window.ClearMousePointer = &@gui_window_clear_mouse_pointer; +Gui.Window.SetOpacity = &@gui_window_set_opacity; +Gui.Window.SetTitle = &@gui_window_set_title; +Gui.Window.SetPosition = &@gui_window_set_position; +Gui.Window.Show = &@gui_window_show; + +"gui "; \ No newline at end of file diff --git a/System/Libraries/Http.HC b/System/Libraries/Http.HC new file mode 100644 index 0000000..8f289f2 --- /dev/null +++ b/System/Libraries/Http.HC @@ -0,0 +1,628 @@ +#define HTTP_TMP_DIRECTORY "B:/Tmp" +#define HTTP_CACHE_DIRECTORY "B:/Tmp/Cache" +#define HTTP_FETCH_BUFFER_SIZE 1024 << 15 + +#define SEDECIM_USER_AGENT_STRING "Mozilla/5.0 (compatible; Sedecim/1.0; TempleOS) (KHTML, like Gecko)" + +class @http_buffer +{ + I64 length; + U8* data; +}; + +class @http_status +{ + U8* protocol; + I64 code; + U8* text; +}; + +class @http_response +{ + I64 state; + TlsSocket* s; + @http_status status; + JsonObject* headers; + @http_buffer body; + Bool headers_parsed; +}; + +class @http_url +{ + U8* scheme; + U8* host; + I64 port; + U8* path; + U8* query; + U8* fragment; +}; + +class @http_request +{ + @http_url* url; + U8* buf; + U8* data; + I64 type; + JsonObject* headers; + @http_response* response; +}; + +#define HttpResponse @http_response +#define HttpUrl @http_url + +#define HTTP_FETCH_BUFFER_SIZE 16777216 + +#define HTTP_PARSE_SCHEME 0 +#define HTTP_PARSE_SCHEME_FS 1 +#define HTTP_PARSE_HOST 2 +#define HTTP_PARSE_PORT 3 +#define HTTP_PARSE_PATH 4 +#define HTTP_PARSE_QUERY 5 +#define HTTP_PARSE_FRAGMENT 6 + +#define HTTP_MIN_REQUEST_BUFFER_SIZE 16384 +#define HTTP_PARSE_URL_FIFO_SIZE 1024 + +#define HTTP_REQ_GET 0 +#define HTTP_REQ_HEAD 1 +#define HTTP_REQ_POST 2 +#define HTTP_REQ_PUT 3 + +#define HTTP_STATE_UNSENT 0 +#define HTTP_STATE_OPENED 1 +#define HTTP_STATE_HEADERS_RECEIVED 2 +#define HTTP_STATE_LOADING 3 +#define HTTP_STATE_DONE 4 + +U8* @http_string_from_fifo(CFifoU8* f) +{ + U8 ch; + I64 i = 0; + U8* str = CAlloc(FifoU8Cnt(f) + 1, erythros_mem_task); + while (FifoU8Cnt(f)) { + FifoU8Rem(f, &ch); + str[i] = ch; + i++; + } + FifoU8Flush(f); + return str; +} + +U0 @http_free_url(@http_url* url) +{ + if (!url) + return; + if (url->scheme) + Free(url->scheme); + if (url->host) + Free(url->host); + if (url->path) + Free(url->path); + if (url->query) + Free(url->query); + if (url->fragment) + Free(url->fragment); + Free(url); +} + +U0 @http_free_response(@http_response* resp) +{ + if (!resp) + return; + // FIXME: Free response headers JSON object + Free(resp); +} + +@http_url* @http_parse_url(U8* str) +{ + if (!str) + return NULL; + U8* buf = NULL; + U8 hex[3]; + I64 i = 0; + I64 state = HTTP_PARSE_SCHEME; + CFifoU8* consume_fifo = FifoU8New(HTTP_PARSE_URL_FIFO_SIZE, erythros_mem_task); + @http_url* url = CAlloc(sizeof(@http_url), erythros_mem_task); + while (1) { + switch (str[i]) { + case 0: + switch (state) { + case HTTP_PARSE_HOST: + url->host = @http_string_from_fifo(consume_fifo); + url->path = StrNew("/", erythros_mem_task); + goto done_parsing_url; + break; + case HTTP_PARSE_PORT: + buf = @http_string_from_fifo(consume_fifo); + url->port = Str2I64(buf); + Free(buf); + url->path = StrNew("/", erythros_mem_task); + goto done_parsing_url; + break; + case HTTP_PARSE_PATH: + url->path = @http_string_from_fifo(consume_fifo); + goto done_parsing_url; + break; + case HTTP_PARSE_QUERY: + url->query = @http_string_from_fifo(consume_fifo); + goto done_parsing_url; + break; + case HTTP_PARSE_FRAGMENT: + url->fragment = @http_string_from_fifo(consume_fifo); + goto done_parsing_url; + break; + default: + goto done_parsing_url; + break; + } + break; + case '#': + switch (state) { + case HTTP_PARSE_PATH: + url->path = @http_string_from_fifo(consume_fifo); + FifoU8Ins(consume_fifo, str[i]); + state = HTTP_PARSE_FRAGMENT; + break; + case HTTP_PARSE_QUERY: + url->query = @http_string_from_fifo(consume_fifo); + FifoU8Ins(consume_fifo, str[i]); + state = HTTP_PARSE_FRAGMENT; + break; + } + break; + case '?': + switch (state) { + case HTTP_PARSE_PATH: + url->path = @http_string_from_fifo(consume_fifo); + FifoU8Ins(consume_fifo, str[i]); + state = HTTP_PARSE_QUERY; + break; + } + break; + case '/': + switch (state) { + case HTTP_PARSE_SCHEME: + state = HTTP_PARSE_SCHEME_FS; + goto keep_consuming_url_chars; + break; + case HTTP_PARSE_SCHEME_FS: + FifoU8Ins(consume_fifo, str[i]); + url->scheme = @http_string_from_fifo(consume_fifo); + if (!StrCmp(url->scheme, "http://")) + url->port = 80; + if (!StrCmp(url->scheme, "https://")) + url->port = 443; + state = HTTP_PARSE_HOST; + break; + case HTTP_PARSE_HOST: + url->host = @http_string_from_fifo(consume_fifo); + FifoU8Ins(consume_fifo, str[i]); + state = HTTP_PARSE_PATH; + break; + case HTTP_PARSE_PORT: + buf = @http_string_from_fifo(consume_fifo); + url->port = Str2I64(buf); + Free(buf); + FifoU8Ins(consume_fifo, str[i]); + state = HTTP_PARSE_PATH; + break; + case HTTP_PARSE_PATH: + goto keep_consuming_url_chars; + break; + } + break; + case ':': + switch (state) { + case HTTP_PARSE_SCHEME: + case HTTP_PARSE_PATH: + case HTTP_PARSE_QUERY: + case HTTP_PARSE_FRAGMENT: + goto keep_consuming_url_chars; + break; + case HTTP_PARSE_HOST: + url->host = @http_string_from_fifo(consume_fifo); + state = HTTP_PARSE_PORT; + break; + } + break; + default: + keep_consuming_url_chars: + switch (state) { + case HTTP_PARSE_PATH: + case HTTP_PARSE_QUERY: + switch (str[i]) { + case '0' ... '9': + case 'A' ... 'Z': + case 'a' ... 'z': + case '?': + case '&': + case '/': + case '=': + // !'()*-._~ + case '!': + case '\'': + case '(': + case ')': + case '*': + case '-': + case '.': + case '_': + case '~': + case '%': + FifoU8Ins(consume_fifo, str[i]); + break; + default: + FifoU8Ins(consume_fifo, '%'); + StrPrint(hex, "%02X", str[i]); + FifoU8Ins(consume_fifo, hex[0]); + FifoU8Ins(consume_fifo, hex[1]); + break; + } + break; + default: + FifoU8Ins(consume_fifo, str[i]); + break; + } + break; + } + i++; + } +done_parsing_url: + FifoU8Flush(consume_fifo); + FifoU8Del(consume_fifo); + return url; +} + +U0 @http_parse_response_headers(@http_response* resp, U8* buffer, I64 length) +{ + if (!resp || !buffer || !length) + return; + U64 response_data_ptr = StrFind("\r\n\r\n", buffer); + if (!response_data_ptr) + return; + resp->body.data = response_data_ptr + 4; + resp->body.data[-4] = NULL; + JsonObject* headers = Json.CreateObject(erythros_mem_task); + U8** lines = NULL; + I64 lines_count = 0; + I64 i; + I64 j; + U8* key_ptr = NULL; + U8* value_ptr = NULL; + lines = String.Split(buffer, '\n', &lines_count); + + U8* resp_protocol = lines[0]; + U8* resp_status_code = StrFind(" ", resp_protocol) + 1; + U8* resp_text = StrFind(" ", resp_status_code + 1); + (*StrFind(" ", resp_protocol)) = NULL; + (*StrFind(" ", resp_status_code)) = NULL; + + resp->status.protocol = StrNew(resp_protocol, erythros_mem_task); + resp->status.code = Str2I64(resp_status_code); + resp->status.text = StrNew(resp_text, erythros_mem_task); + + for (i = 1; i < lines_count; i++) { + for (j = 0; j < StrLen(lines[i]); j++) { + if (lines[i][j] == ':' && lines[i][j + 1] == ' ') { + lines[i][j] = NULL; + key_ptr = lines[i]; + value_ptr = lines[i] + j + 2; + (*StrFind("\r", value_ptr)) = NULL; + headers->set(key_ptr, value_ptr, JSON_STRING); + goto @http_next_header_line; + } + } + @http_next_header_line: + } + resp->headers = headers; + resp->headers_parsed = TRUE; +} + +Bool @http_detect_response_headers(U8* buf, I64 len) +{ + if (len < 4) + return FALSE; + I64 i; + for (i = 0; i < len - 4; i++) { + if (!MemCmp(buf + i, "\r\n\r\n", 4)) + return TRUE; + } + return FALSE; +} + +I64 @http_req(@http_request* req) +{ + if (!req) + return NULL; + if (!req->url || !req->buf || !req->response) + return NULL; + if (!req->url->scheme || !req->url->host || !req->url->path) + return NULL; + if (req->type == HTTP_REQ_POST && !req->data) + return NULL; + if (req->type == HTTP_REQ_PUT && !req->data) + return NULL; + + @http_response* resp = req->response; + + U8* buf = NULL; + U8* headers_buf = ""; + I64 cnt = 1; + I64 len = NULL; + + buf = CAlloc(HTTP_MIN_REQUEST_BUFFER_SIZE, erythros_mem_task); + if (req->headers) { + headers_buf = CAlloc(HTTP_MIN_REQUEST_BUFFER_SIZE, erythros_mem_task); + JsonKey* key = req->headers->keys; + while (key) { + StrPrint(headers_buf + StrLen(headers_buf), "%s: %s\r\n", key->name, key->value); + key = key->next; + } + } + + switch (req->type) { + case HTTP_REQ_GET: + StrPrint(buf, + "GET %s%s HTTP/1.0\r\n" + "Host: %s\r\n" + "%s" + "User-Agent: " SEDECIM_USER_AGENT_STRING + "\r\n\r\n", + req->url->path, req->url->query, req->url->host, headers_buf); + break; + case HTTP_REQ_HEAD: + StrPrint(buf, + "HEAD %s%s HTTP/1.0\r\n" + "Host: %s\r\n" + "%s" + "User-Agent: " SEDECIM_USER_AGENT_STRING + "\r\n\r\n", + req->url->path, req->url->query, req->url->host, headers_buf); + break; + case HTTP_REQ_POST: + StrPrint(buf, + "POST %s%s HTTP/1.0\r\n" + "Host: %s\r\n" + "%s" + "User-Agent: " SEDECIM_USER_AGENT_STRING + "\r\n" + "Content-Length: %d\r\n\r\n", + req->url->path, req->url->query, req->url->host, headers_buf, + StrLen(req->data)); + StrPrint(buf + StrLen(buf), req->data); + break; + case HTTP_REQ_PUT: + StrPrint(buf, + "PUT %s%s HTTP/1.0\r\n" + "Host: %s\r\n" + "%s" + "User-Agent: " SEDECIM_USER_AGENT_STRING + "\r\n" + "Content-Length: %d\r\n\r\n", + req->url->path, req->url->query, req->url->host, headers_buf, + StrLen(req->data)); + StrPrint(buf + StrLen(buf), req->data); + break; + } + + TlsSocket* s = NULL; + resp->s = NULL; + + if (!StrCmp(req->url->scheme, "http://")) { + s = @tcp_socket_create(req->url->host, req->url->port); + resp->s = s; + while (s->state != TCP_SOCKET_STATE_ESTABLISHED) + Sleep(1); + } + + if (!StrCmp(req->url->scheme, "https://")) { + s = @tls_socket_create(req->url->host, req->url->port); + resp->s = s; + while (!@tls_established(s->ctx)) + Sleep(1); + } + + resp->state = HTTP_STATE_OPENED; + s->send(buf, StrLen(buf)); + while (cnt || s->state != TCP_SOCKET_STATE_CLOSED) { + cnt = s->receive(req->buf + len, 1024); + len += cnt; + switch (resp->state) { + case HTTP_STATE_LOADING: + resp->body.length += cnt; + break; + case HTTP_STATE_HEADERS_RECEIVED: + resp->body.length = (req->buf + len) - resp->body.data; + resp->state = HTTP_STATE_LOADING; + break; + case HTTP_STATE_OPENED: + if (@http_detect_response_headers(req->buf, len)) { + @http_parse_response_headers(resp, req->buf, len); + resp->state = HTTP_STATE_HEADERS_RECEIVED; + } + break; + } + Sleep(1); + } + if (!resp->headers_parsed) + @http_parse_response_headers(resp, req->buf, len); + resp->state = HTTP_STATE_DONE; + req->buf[len] = NULL; + Free(buf); + if (StrLen(headers_buf) > 0) { + Free(headers_buf); + } + s->close(); + resp->s = NULL; + Free(req); + return len; +} + +Bool @http_scheme_is_https(@http_url* url) +{ + if (!url || !url->scheme) + return FALSE; + return !MemCmp(url->scheme, "https", 5); +} + +@http_response* @http_get(@http_url* url, U8* buf, U64 return_req = NULL, JsonObject* headers = NULL) +{ + @http_response* resp = CAlloc(sizeof(@http_response), erythros_mem_task); + @http_request* req = CAlloc(sizeof(@http_request), erythros_mem_task); + if (return_req) + MemCpy(return_req, &req, sizeof(U64)); + req->url = url; + req->buf = buf; + req->type = HTTP_REQ_GET; + req->headers = headers; + req->response = resp; + Spawn(&@http_req, req, "HTTPGetRequest"); + return resp; +} + +@http_response* @http_head(@http_url* url, U8* buf, JsonObject* headers = NULL) +{ + @http_response* resp = CAlloc(sizeof(@http_response), erythros_mem_task); + @http_request* req = CAlloc(sizeof(@http_request), erythros_mem_task); + req->url = url; + req->buf = buf; + req->type = HTTP_REQ_HEAD; + req->headers = headers; + req->response = resp; + Spawn(&@http_req, req, "HTTPHeadRequest"); + return resp; +} + +@http_response* @http_post(@http_url* url, U8* buf, U8* data, JsonObject* headers = NULL) +{ + @http_response* resp = CAlloc(sizeof(@http_response), erythros_mem_task); + @http_request* req = CAlloc(sizeof(@http_request), erythros_mem_task); + req->url = url; + req->buf = buf; + req->type = HTTP_REQ_POST; + req->headers = headers; + req->data = data; + req->response = resp; + Spawn(&@http_req, req, "HTTPPostRequest"); + return resp; +} + +@http_response* @http_put(@http_url* url, U8* buf, U8* data, JsonObject* headers = NULL) +{ + @http_response* resp = CAlloc(sizeof(@http_response), erythros_mem_task); + @http_request* req = CAlloc(sizeof(@http_request), erythros_mem_task); + req->url = url; + req->buf = buf; + req->type = HTTP_REQ_PUT; + req->headers = headers; + req->data = data; + req->response = resp; + Spawn(&@http_req, req, "HTTPPutRequest"); + return resp; +} + +class @http +{ + @http_response* (*Get)(@http_url* url, U8* buf, U64* req = NULL, JsonObject* headers = NULL); + @http_response* (*Head)(@http_url* url, U8* buf, JsonObject* headers = NULL); + @http_response* (*Post)(@http_url* url, U8* buf, U8* data, JsonObject* headers = NULL); + @http_response* (*Put)(@http_url* url, U8* buf, U8* data, JsonObject* headers = NULL); +}; + +@http Http; + +Http.Get = &@http_get; +Http.Head = &@http_head; +Http.Post = &@http_post; +Http.Put = &@http_put; + +Bool @http_is_resource_cached(U8* src, U8* cache_directory = HTTP_CACHE_DIRECTORY) +{ + U8 buf[512]; + U8* src_md5 = md5_string(src, StrLen(src)); + StrCpy(buf, cache_directory); + StrPrint(buf + StrLen(buf), "/%s", src_md5); + Free(src_md5); + return FileFind(buf); +} + +U0 @http_cache_resource(U8* src, U8* data, I64 size, U8* cache_directory = HTTP_CACHE_DIRECTORY) +{ + U8 buf[512]; + U8* src_md5 = md5_string(src, StrLen(src)); + StrCpy(buf, cache_directory); + StrPrint(buf + StrLen(buf), "/%s", src_md5); + Free(src_md5); + FileWrite(buf, data, size); +} + +U8* @http_get_cached_resource_filename(U8* src, U8* cache_directory = HTTP_CACHE_DIRECTORY) +{ + U8* buf = CAlloc(512, erythros_mem_task); + U8* src_md5 = md5_string(src, StrLen(src)); + StrCpy(buf, cache_directory); + StrPrint(buf + StrLen(buf), "/%s", src_md5); + Free(src_md5); + return buf; +} + +U0 @http_init_tmp_and_cache_directories() +{ + if (!FileFind(HTTP_TMP_DIRECTORY)) + DirMk(HTTP_TMP_DIRECTORY); + if (!FileFind(HTTP_CACHE_DIRECTORY)) + DirMk(HTTP_CACHE_DIRECTORY); +} + +@http_response* fetch(U8* url_string, U8* fetch_buffer) +{ + if (!url_string || !fetch_buffer) + return NULL; + HttpUrl* url = @http_parse_url(url_string); + if (!url) + return NULL; + @http_response* resp = Http.Get(url, fetch_buffer); + while (resp->state != HTTP_STATE_DONE) + Sleep(1); + return resp; +} + +I64 curl(U8* url_string) +{ + if (!url_string) + return 0; + U8* fetch_buffer = CAlloc(HTTP_FETCH_BUFFER_SIZE, erythros_mem_task); + @http_response* resp = fetch(url_string, fetch_buffer); + if (!resp) + return 0; + if (resp->body.length) { + U8* buf = resp->body.data; + while (*buf) { + if (*buf == '\d') + "\d\d"; + else + "%c", *buf; + ++buf; + } + "\n"; + } + Free(fetch_buffer); + return resp->body.length; +} + +I64 download(U8* path, U8* url_string) +{ + if (!path || !url_string) + return 0; + U8* fetch_buffer = CAlloc(HTTP_FETCH_BUFFER_SIZE, erythros_mem_task); + @http_response* resp = fetch(url_string, fetch_buffer); + if (!resp) + return 0; + if (resp->body.length) { + FileWrite(path, resp->body.data, resp->body.length); + } + Free(fetch_buffer); + return resp->body.length; +} + +"http "; diff --git a/System/Libraries/Image.HC b/System/Libraries/Image.HC new file mode 100644 index 0000000..09d172e --- /dev/null +++ b/System/Libraries/Image.HC @@ -0,0 +1,55 @@ +class @image +{ + Context2D* (*FileToContext2D)(U8* filepath); + Context2D* (*BufferToContext2D)(U8* buffer, I64 size); +}; + +@image Image; + +Context2D* @image_buffer_to_context2d(U8* buffer, I64 size) +{ + if (!buffer || !size) { + return NULL; + } + I32 x; + I32 y; + I32 comp; + I32 code = @stbi_info_from_memory(buffer, size, &x, &y, &comp); + if (code != 1) { + return NULL; + } + U8* pixels = @stbi_load_from_memory(buffer, size, &x, &y, &comp, 4); + if (!pixels) { + return NULL; + } + Context2D* ctx = CAlloc(sizeof(Context2D)); + ctx->width = x; + ctx->height = y; + ctx->bpp = 32; + ctx->fb = pixels; + I64 i; + for (i = 0; i < x * y; i++) { + ctx->fb(U32*)[i] = @image_pixel_flip_rgb_bgr(ctx->fb(U32*)[i]); + } + return ctx; +} + +Context2D* @image_file_to_context2d(U8* filepath) +{ + if (!FileFind(filepath)) { + return NULL; + } + I64 size = NULL; + U8* buffer = FileRead(filepath, &size); + if (!buffer || !size) { + return NULL; + } + Context2D* ctx = @image_buffer_to_context2d(buffer, size); + Free(buffer); + return ctx; +} + +Image.FileToContext2D = &@image_file_to_context2d; +Image.BufferToContext2D = &@image_buffer_to_context2d; + +"image "; diff --git a/System/Libraries/Ipc.HC b/System/Libraries/Ipc.HC new file mode 100644 index 0000000..cf6550a --- /dev/null +++ b/System/Libraries/Ipc.HC @@ -0,0 +1,54 @@ +#define ipc user_data.u32[0] +#define IPC_QUEUE_SIZE 1024 +#define IPC_ENQUEUE_LIMIT 16 + +class IpcMessage { + CTask* client; + I64 type; + U64 payload; + I64 i64; +}; + +class @ipc +{ + U0 (*InitQueue)(CTask* task); + IpcMessage* (*MsgRecv)(); + U0 (*MsgSend)(CTask* task, IpcMessage* msg); +}; + +U0 @ipc_queue_init(CTask* task) +{ // Initialize a task's IpcMessage Queue + MemSetU32(task->pad, 0, 1); + if (!task->ipc) { + task->ipc = FifoI64New(IPC_QUEUE_SIZE, task->code_heap); + } +} + +IpcMessage* @ipc_msg_recv() +{ // Receive a IpcMessage from current task's + // message queue. + U64 msg_ptr; + if (!FifoI64Cnt(Fs->ipc)) + return FALSE; + FifoI64Rem(Fs->ipc, &msg_ptr); + return msg_ptr; +} + +U0 @ipc_msg_send( + CTask* task, + IpcMessage* msg) +{ // Send a IpcMessage to a task (client or server) + if (FifoI64Cnt(task->ipc) > IPC_ENQUEUE_LIMIT) { + Free(msg); + return; + } + FifoI64Ins(task->ipc, msg); +} + +@ipc Ipc; + +Ipc.InitQueue = &@ipc_queue_init; +Ipc.MsgRecv = &@ipc_msg_recv; +Ipc.MsgSend = &@ipc_msg_send; + +"ipc "; diff --git a/System/Libraries/Json.HC b/System/Libraries/Json.HC new file mode 100644 index 0000000..4d432fb --- /dev/null +++ b/System/Libraries/Json.HC @@ -0,0 +1,1550 @@ +#define JSON_SAME -1 +#define JSON_UNDEFINED 0 +#define JSON_OBJECT 1 +#define JSON_ARRAY 2 +#define JSON_STRING 3 +#define JSON_NUMBER 4 +#define JSON_BOOLEAN 5 +#define JSON_NULL 6 +#define JSON_HTML 7 + +#define JSON_ELEMENT_IS_KEY 1 +#define JSON_ELEMENT_IS_ITEM 2 + +#define JSON_STATE_OBJECT_OR_ARRAY 0 + +#define JSON_STATE_OBJECT 100 +#define JSON_STATE_OBJECT_KEY 101 +#define JSON_STATE_OBJECT_SEPARATOR 102 +#define JSON_STATE_OBJECT_TYPE 103 +#define JSON_STATE_OBJECT_NEXT 104 + +#define JSON_STATE_OBJECT_OBJECT 105 +#define JSON_STATE_OBJECT_ARRAY 106 +#define JSON_STATE_OBJECT_STRING 107 +#define JSON_STATE_OBJECT_NUMBER 108 +#define JSON_STATE_OBJECT_BOOLEAN 109 +#define JSON_STATE_OBJECT_NULL 110 + +#define JSON_STATE_ARRAY 200 +#define JSON_STATE_ARRAY_TYPE 201 +#define JSON_STATE_ARRAY_NEXT 202 + +#define JSON_STATE_ARRAY_OBJECT 203 +#define JSON_STATE_ARRAY_ARRAY 204 +#define JSON_STATE_ARRAY_STRING 205 +#define JSON_STATE_ARRAY_NUMBER 206 +#define JSON_STATE_ARRAY_BOOLEAN 207 +#define JSON_STATE_ARRAY_NULL 208 + +#define JSON_PARSER_FIFO_SIZE 32768 + +#define JSON_WRAPPER_MAGIC_NUMBER 0xDEADC0DEDEADC0DE + +#define JSON_SIG 0xFABACEAE + +class @json_stringify_string +{ + I64 length; + U8* value; + I64 capacity; + CTask* mem_task; +} + +class @json_element +{ + U32 sig; + @json_element* prev; + @json_element* next; + I64 type; +}; + +class @json_key : @json_element +{ + U8* name; + U64 value; +}; + +class @json_item : @json_element { U64 value; }; + +class @json_object : @json_element +{ + I64 length; + @json_key* keys; + CTask* mem_task; +}; + +class @json_array : @json_element +{ + I64 length; + @json_item* items; + @json_item* last_item; + CTask* mem_task; +}; + +extern class @json_callable_object; + +class @json_callable_array : @json_array +{ + U64 (*@)(I64 index, Bool return_item = FALSE); + @json_callable_array* (*a)(I64 index, Bool return_item = FALSE); + @json_callable_object* (*o)(I64 index, Bool return_item = FALSE); + U0 (*append)(U64 value, I64 type = NULL); + Bool (*contains)(U64 value, I64 type = NULL, Bool match_case = FALSE); + U0 (*insert)(I64 index, U64 value, I64 type = NULL); + U0 (*prepend)(U64 value, I64 type = NULL); + U0 (*remove)(I64 index); +}; + +class @json_callable_object : @json_object +{ + U64 (*@)(U8* key, Bool return_key = FALSE); + @json_callable_array* (*a)(U8* key, Bool return_key = FALSE); + @json_callable_object* (*o)(U8* key, Bool return_key = FALSE); + U0 (*set)(U8* key, U64 value, I64 type = JSON_SAME); + U0 (*unset)(U8* key); +}; + +class @json_parser +{ + U8* stream; + U8 token; + CFifoU8* consumed; + I64 pos; + I64 state; + Bool debug; + CTask* mem_task; +}; + +#define JsonArray @json_callable_array +#define JsonElement @json_element +#define JsonItem @json_item +#define JsonKey @json_key +#define JsonObject @json_callable_object + +U0 @json_debug_parser_state(@json_parser* parser) +{ + switch (parser->state) { + case JSON_STATE_OBJECT: + "JSON_STATE_OBJECT\n"; + break; + case JSON_STATE_OBJECT_KEY: + "JSON_STATE_OBJECT_KEY\n"; + break; + case JSON_STATE_OBJECT_SEPARATOR: + "JSON_STATE_OBJECT_SEPARATOR\n"; + break; + case JSON_STATE_OBJECT_TYPE: + "JSON_STATE_OBJECT_TYPE\n"; + break; + case JSON_STATE_OBJECT_NEXT: + "JSON_STATE_OBJECT_NEXT\n"; + break; + case JSON_STATE_OBJECT_STRING: + "JSON_STATE_OBJECT_STRING\n"; + break; + case JSON_STATE_OBJECT_NUMBER: + "JSON_STATE_OBJECT_NUMBER\n"; + break; + case JSON_STATE_ARRAY: + "JSON_STATE_ARRAY\n"; + break; + case JSON_STATE_ARRAY_TYPE: + "JSON_STATE_ARRAY_TYPE\n"; + break; + case JSON_STATE_ARRAY_NEXT: + "JSON_STATE_ARRAY_NEXT\n"; + break; + case JSON_STATE_ARRAY_STRING: + "JSON_STATE_ARRAY_STRING\n"; + break; + case JSON_STATE_ARRAY_NUMBER: + "JSON_STATE_ARRAY_NUMBER\n"; + break; + } +} + +@json_item* @json_create_item(U64 value, I64 type = NULL, CTask* mem_task) +{ + @json_item* item = CAlloc(sizeof(@json_item), mem_task); + item->sig = JSON_SIG; + item->type = type; + if (!item->type) { + if (value(@json_element*)->sig == JSON_SIG) { + item->type = value(@json_element*)->type; + } else if (value > 0x1000) { + item->type = JSON_STRING; + } else { + item->type = JSON_BOOLEAN; + value = value != 0; + } + } + if (item->type == JSON_STRING) + item->value = StrNew(value, mem_task); + else + item->value = value; + return item; +} + +U0 @json_append_item(@json_array* arr, U64 value, I64 type = NULL) +{ + if (!arr) + return; + if (arr->type != JSON_ARRAY) + return; + @json_item* append_item = @json_create_item(value, type, arr->mem_task); + @json_item* item = arr->last_item; + if (!item) { + append_item->prev = NULL; + append_item->next = NULL; + arr->items = append_item; + arr->last_item = append_item; + arr->length++; + return; + } + item->next = append_item; + append_item->prev = item; + arr->last_item = append_item; + arr->length++; +} + +U8* @json_string_from_fifo(CFifoU8* f, CTask* mem_task) +{ + U8 ch; + I64 i = 0; + U8* str = CAlloc(FifoU8Cnt(f) + 1, mem_task); + while (FifoU8Cnt(f)) { + FifoU8Rem(f, &ch); + str[i] = ch; + i++; + } + FifoU8Flush(f); + return str; +} + +U0 @json_insert_key(@json_object* obj, @json_key* key) +{ + if (!obj) + return; + if (!obj->keys) { + obj->keys = key; + obj->length++; + return; + } + @json_key* k = obj->keys; + while (k->next) + k = k->next; + k->next = key; + key->prev = k; + obj->length++; +} + +U0 @json_rstrip(U8* str) +{ + if (!str || !StrLen(str)) + return; + I64 r_pos = StrLen(str) - 1; + while (str[r_pos] == ' ' || str[r_pos] == '\r' || str[r_pos] == '\n' || str[r_pos] == '\t') { + str[r_pos] = NULL; + --r_pos; + } +} + +extern @json_element* @json_parse_object_or_array(@json_parser* parser); + +I64 @json_unescape_char(I64 escape_ch) +{ + I64 return_ch = escape_ch; + + // FIXME: unicode + switch (escape_ch) { + case 'b': + return_ch = 0x08; + break; + case 'f': + return_ch = 0x0c; + break; + case 'n': + return_ch = 0x0a; + break; + case 'r': + return_ch = 0x0d; + break; + case 't': + return_ch = 0x09; + break; + default: + break; + } + + return return_ch; +} + +U0 @json_parse_object(@json_parser* parser, @json_object* obj) +{ + @json_key* key = NULL; + while (1) { + switch (parser->stream[parser->pos]) { + case '\\': + // NOTE: We keep escaped unicode in its original form, and let the program ingesting the JSON handle the UTF-8 conversion. + if (parser->state == JSON_STATE_OBJECT_STRING) { + if (parser->stream[parser->pos + 1] == 'u') { + FifoU8Ins(parser->consumed, '\\'); + } else { + FifoU8Ins(parser->consumed, @json_unescape_char(parser->stream[++parser->pos])); + } + } + break; + case '}': + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_NUMBER: + key->value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(key->value); + key->value = Str2F64(key->value); + @json_insert_key(obj, key); + return; + break; + case JSON_STATE_OBJECT_BOOLEAN: + key->value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(key->value); + if (StrCmp("true", key->value) && StrCmp("false", key->value)) { + PrintErr("@json_parse_object: Illegal boolean value at position %d", + parser->pos); + while (1) + Sleep(1); + } + if (!StrCmp("true", key->value)) + key->value = TRUE; + else + key->value = FALSE; + @json_insert_key(obj, key); + return; + break; + case JSON_STATE_OBJECT_NULL: + key->value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(key->value); + if (StrCmp("null", key->value)) { + PrintErr("@json_parse_object: Illegal null value at position %d", + parser->pos); + while (1) + Sleep(1); + } + key->value = NULL; + @json_insert_key(obj, key); + return; + break; + case JSON_STATE_OBJECT: + case JSON_STATE_OBJECT_NEXT: + return; + break; + } + break; + case ',': + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_NUMBER: + key->value = @json_string_from_fifo(parser->consumed, parser->mem_task); + key->value = Str2F64(key->value); + @json_insert_key(obj, key); + parser->state = JSON_STATE_OBJECT; + break; + case JSON_STATE_OBJECT_BOOLEAN: + key->value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(key->value); + if (StrCmp("true", key->value) && StrCmp("false", key->value)) { + PrintErr("@json_parse_object: Illegal boolean value at position %d", + parser->pos); + while (1) + Sleep(1); + } + if (!StrCmp("true", key->value)) + key->value = TRUE; + else + key->value = FALSE; + @json_insert_key(obj, key); + parser->state = JSON_STATE_OBJECT; + break; + case JSON_STATE_OBJECT_NULL: + key->value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(key->value); + if (StrCmp("null", key->value)) { + PrintErr("@json_parse_object: Illegal null value at position %d", + parser->pos); + while (1) + Sleep(1); + } + key->value = NULL; + @json_insert_key(obj, key); + parser->state = JSON_STATE_OBJECT; + break; + case JSON_STATE_OBJECT_NEXT: + parser->state = JSON_STATE_OBJECT; + break; + } + break; + case ':': + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_SEPARATOR: + parser->state = JSON_STATE_OBJECT_TYPE; + break; + } + break; + case '[': + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_TYPE: + key->type = JSON_ARRAY; + key->value = @json_parse_object_or_array(parser); + @json_insert_key(obj, key); + parser->state = JSON_STATE_OBJECT_NEXT; + break; + } + break; + case '{': + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_TYPE: + key->type = JSON_OBJECT; + key->value = @json_parse_object_or_array(parser); + @json_insert_key(obj, key); + parser->state = JSON_STATE_OBJECT_NEXT; + break; + } + break; + case '"': + switch (parser->state) { + case JSON_STATE_OBJECT_STRING: + key->value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_insert_key(obj, key); + parser->state = JSON_STATE_OBJECT_NEXT; + break; + case JSON_STATE_OBJECT_TYPE: + key->type = JSON_STRING; + parser->state = JSON_STATE_OBJECT_STRING; + break; + case JSON_STATE_OBJECT_KEY: + key->name = @json_string_from_fifo(parser->consumed, parser->mem_task); + parser->state = JSON_STATE_OBJECT_SEPARATOR; + break; + case JSON_STATE_OBJECT: + key = CAlloc(sizeof(@json_key), parser->mem_task); + key->sig = JSON_SIG; + parser->state = JSON_STATE_OBJECT_KEY; + break; + } + break; + case '-': + case '0' ... '9': + case '.': + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + case JSON_STATE_OBJECT_NUMBER: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_TYPE: + key->type = JSON_NUMBER; + parser->state = JSON_STATE_OBJECT_NUMBER; + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + } + break; + case 't': + case 'f': + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_TYPE: + key->type = JSON_BOOLEAN; + parser->state = JSON_STATE_OBJECT_BOOLEAN; + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + } + break; + case 'n': + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_TYPE: + key->type = JSON_NULL; + parser->state = JSON_STATE_OBJECT_NULL; + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + } + break; + default: + switch (parser->state) { + case JSON_STATE_OBJECT_KEY: + case JSON_STATE_OBJECT_STRING: + case JSON_STATE_OBJECT_BOOLEAN: + case JSON_STATE_OBJECT_NULL: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + } + break; + } + if (parser->debug) { + @json_debug_parser_state(parser); + "Object: %08X, Pos: %d, Token: %c\n", obj, parser->pos, + parser->stream[parser->pos]; + Sleep(50); + } + parser->pos++; + } +} + +U0 @json_parse_array(@json_parser* parser, @json_array* arr) +{ + U64 value = NULL; + while (1) { + if (parser->state == JSON_STATE_ARRAY) { + switch (parser->stream[parser->pos]) { + case 0: + PrintErr("@json_parse_array: Malformed array"); + while (1) + Sleep(1); + break; + case ']': + return; + break; + } + parser->state = JSON_STATE_ARRAY_TYPE; + } + switch (parser->stream[parser->pos]) { + case '\\': + // NOTE: We keep escaped unicode in its original form, and let the program ingesting the JSON handle the UTF-8 conversion. + if (parser->state == JSON_STATE_ARRAY_STRING) { + if (parser->stream[parser->pos + 1] == 'u') { + FifoU8Ins(parser->consumed, '\\'); + } else { + FifoU8Ins(parser->consumed, @json_unescape_char(parser->stream[++parser->pos])); + } + } + break; + case ']': + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_ARRAY_NUMBER: + value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(value); + value = Str2F64(value); + @json_append_item(arr, value, JSON_NUMBER); + return; + break; + case JSON_STATE_ARRAY_BOOLEAN: + value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(value); + if (StrCmp("true", value) && StrCmp("false", value)) { + PrintErr("@json_parse_array: Illegal boolean value at position %d", + parser->pos); + while (1) + Sleep(1); + } + if (!StrCmp("true", value)) + value = TRUE; + else + value = FALSE; + @json_append_item(arr, value, JSON_BOOLEAN); + break; + case JSON_STATE_ARRAY_NULL: + value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(value); + if (StrCmp("null", value)) { + PrintErr("@json_parse_array: Illegal null value at position %d", + parser->pos); + while (1) + Sleep(1); + } + value = NULL; + @json_append_item(arr, value, JSON_NULL); + break; + case JSON_STATE_ARRAY: + case JSON_STATE_ARRAY_NEXT: + return; + break; + } + break; + case ',': + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_ARRAY_NUMBER: + value = @json_string_from_fifo(parser->consumed, parser->mem_task); + value = Str2F64(value); + @json_append_item(arr, value, JSON_NUMBER); + parser->state = JSON_STATE_ARRAY; + break; + case JSON_STATE_ARRAY_BOOLEAN: + value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(value); + if (StrCmp("true", value) && StrCmp("false", value)) { + PrintErr("@json_parse_array: Illegal boolean value at position %d", + parser->pos); + while (1) + Sleep(1); + } + if (!StrCmp("true", value)) + value = TRUE; + else + value = FALSE; + @json_append_item(arr, value, JSON_BOOLEAN); + parser->state = JSON_STATE_ARRAY; + break; + case JSON_STATE_ARRAY_NULL: + value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_rstrip(value); + if (StrCmp("null", value)) { + PrintErr("@json_parse_array: Illegal null value at position %d", + parser->pos); + while (1) + Sleep(1); + } + value = NULL; + @json_append_item(arr, value, JSON_NULL); + parser->state = JSON_STATE_ARRAY; + break; + case JSON_STATE_ARRAY_NEXT: + parser->state = JSON_STATE_ARRAY; + break; + } + break; + case '[': + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_ARRAY_TYPE: + value = @json_parse_object_or_array(parser); + @json_append_item(arr, value, JSON_ARRAY); + parser->state = JSON_STATE_ARRAY_NEXT; + break; + } + break; + case '{': + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_ARRAY_TYPE: + value = @json_parse_object_or_array(parser); + @json_append_item(arr, value, JSON_OBJECT); + parser->state = JSON_STATE_ARRAY_NEXT; + break; + } + break; + case '"': + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + value = @json_string_from_fifo(parser->consumed, parser->mem_task); + @json_append_item(arr, value, JSON_STRING); + parser->state = JSON_STATE_ARRAY_NEXT; + break; + case JSON_STATE_ARRAY_TYPE: + parser->state = JSON_STATE_ARRAY_STRING; + break; + } + break; + case '-': + case '0' ... '9': + case '.': + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + case JSON_STATE_ARRAY_NUMBER: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_ARRAY_TYPE: + parser->state = JSON_STATE_ARRAY_NUMBER; + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + } + break; + case 't': + case 'f': + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_ARRAY_TYPE: + parser->state = JSON_STATE_ARRAY_BOOLEAN; + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + } + break; + case 'n': + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + case JSON_STATE_OBJECT_TYPE: + parser->state = JSON_STATE_ARRAY_NULL; + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + } + break; + default: + switch (parser->state) { + case JSON_STATE_ARRAY_STRING: + case JSON_STATE_ARRAY_BOOLEAN: + case JSON_STATE_ARRAY_NULL: + FifoU8Ins(parser->consumed, parser->stream[parser->pos]); + break; + } + break; + } + if (parser->debug) { + @json_debug_parser_state(parser); + "Array: %08X, Pos: %d, Token: %c\n", arr, parser->pos, + parser->stream[parser->pos]; + Sleep(50); + } + parser->pos++; + } +} + +extern @json_callable_array* @json_create_callable_array(@json_array* arr); +extern @json_callable_object* @json_create_callable_object(@json_object* obj); + +@json_element* @json_parse_object_or_array(@json_parser* parser) +{ + @json_element* el = CAlloc(sizeof(@json_array), parser->mem_task); + el->sig = JSON_SIG; + while (1) { + switch (parser->stream[parser->pos]) { + case 0: + return el; + break; + case ' ': + case '\r': + case '\n': + case '\t': + break; + case '{': + el->type = JSON_OBJECT; + el(@json_object*)->mem_task = parser->mem_task; + parser->pos++; + parser->state = JSON_STATE_OBJECT; + @json_parse_object(parser, el); + return @json_create_callable_object(el); + break; + case '[': + el->type = JSON_ARRAY; + el(@json_array*)->mem_task = parser->mem_task; + parser->pos++; + parser->state = JSON_STATE_ARRAY; + @json_parse_array(parser, el); + return @json_create_callable_array(el); + break; + default: + PrintErr("@json_parse_object_or_array: Invalid token: '%c'", parser->stream[parser->pos]); + while (1) { + Sleep(1); + }; + break; + } + parser->pos++; + Sleep(1); + } +} + +@json_element* @json_parse(U8* str, CTask* mem_task) +{ + if (!str || !mem_task) { + return NULL; + } + @json_parser* parser = CAlloc(sizeof(@json_parser), mem_task); + parser->mem_task = mem_task; + parser->consumed = FifoU8New(JSON_PARSER_FIFO_SIZE, parser->mem_task); + parser->stream = str; + return @json_parse_object_or_array(parser); +} + +U0 @json_prepend_item(@json_array* arr, U64 value, I64 type) +{ + if (!arr) + return; + if (arr->type != JSON_ARRAY) + return; + @json_item* prepend_item = @json_create_item(value, type, arr->mem_task); + @json_item* items = arr->items; + arr->items = prepend_item; + arr->items->next = items; + arr->length++; +} + +U0 @json_insert_item(@json_array* arr, I64 index, U64 value, I64 type) +{ + if (!arr) + return; + if (arr->type != JSON_ARRAY) + return; + if (index <= 0) { + @json_prepend_item(arr, value, type); + return; + } + if (index >= arr->length) { + @json_append_item(arr, value, type); + return; + } + @json_item* insert_item = @json_create_item(value, type, arr->mem_task); + @json_item* insert_at_item = arr->items; + @json_item* insert_after_item = NULL; + I64 i; + for (i = 0; i < index; i++) + insert_at_item = insert_at_item->next; + insert_after_item = insert_at_item->prev; + insert_after_item->next = insert_item; + insert_item->prev = insert_after_item; + insert_item->next = insert_at_item; + insert_at_item->prev = insert_item; + arr->length++; +} + +U0 @json_stringify_check_capacity(@json_stringify_string* str) +{ + if (str->length >= str->capacity) { + str->capacity *= 2; + U8* new_value = CAlloc(str->capacity * 2, str->mem_task); + MemCpy(new_value, str->value, str->length); + str->value = new_value; + } +} + +U0 @json_stringify_append_char(@json_stringify_string* str, U8 char) +{ + // FIXME: unicode + switch (char) { + case '\\': + str->value[str->length++] = '\\'; + str->value[str->length++] = '\\'; + break; + case 0x08: + str->value[str->length++] = '\\'; + str->value[str->length++] = 'b'; + break; + case 0x0c: + str->value[str->length++] = '\\'; + str->value[str->length++] = 'f'; + break; + case 0x0a: + str->value[str->length++] = '\\'; + str->value[str->length++] = 'n'; + break; + case 0x0d: + str->value[str->length++] = '\\'; + str->value[str->length++] = 'r'; + break; + case 0x09: + str->value[str->length++] = '\\'; + str->value[str->length++] = 't'; + break; + default: + str->value[str->length++] = char; + break; + } + str->value[str->length] = NULL; + @json_stringify_check_capacity(str); +} + +U0 @json_stringify_append_char_escape_quotes(@json_stringify_string* str, U8 char) +{ + // FIXME: unicode + switch (char) { + case '"': + str->value[str->length++] = '\\'; + str->value[str->length++] = '"'; + break; + case '\\': + str->value[str->length++] = '\\'; + str->value[str->length++] = '\\'; + break; + case 0x08: + str->value[str->length++] = '\\'; + str->value[str->length++] = 'b'; + break; + case 0x0c: + str->value[str->length++] = '\\'; + str->value[str->length++] = 'f'; + break; + case 0x0a: + str->value[str->length++] = '\\'; + str->value[str->length++] = 'n'; + break; + case 0x0d: + str->value[str->length++] = '\\'; + str->value[str->length++] = 'r'; + break; + case 0x09: + str->value[str->length++] = '\\'; + str->value[str->length++] = 't'; + break; + default: + str->value[str->length++] = char; + break; + } + str->value[str->length] = NULL; + @json_stringify_check_capacity(str); +} + +U0 @json_stringify_append_str(@json_stringify_string* str, U8* str2) +{ + while (*str2) { + // NOTE: We keep escaped unicode in its original form, and let the program ingesting the JSON handle the UTF-8 conversion. + if (*str2 == '\\' && *(str2 + 1) == 'u') { + str->value[str->length++] = '\\'; + str->value[str->length++] = 'u'; + str->value[str->length] = NULL; + str2++; + } else { + @json_stringify_append_char_escape_quotes(str, *str2); + } + str2++; + @json_stringify_check_capacity(str); + } +} + +U0 @json_stringify_append_number(@json_stringify_string* str, F64 num) +{ + U8 buf[256]; + MemSet(buf, 0, 256); + StrPrint(buf, "%.6f", num); + I64 i = StrLen(buf) - 1; + while (buf[i] == '0') { + buf[i] = NULL; + i--; + } + i = StrLen(buf) - 1; + if (buf[i] == '.') + buf[i] = NULL; + @json_stringify_append_str(str, buf); +} + +extern U0 @json_stringify_object_or_array(@json_stringify_string* str, @json_element* el); + +U0 @json_stringify_object(@json_stringify_string* str, @json_object* obj) +{ + @json_stringify_append_char(str, '{'); + @json_key* key = obj->keys; + while (key) { + @json_stringify_append_char(str, '"'); + @json_stringify_append_str(str, key->name); + @json_stringify_append_char(str, '"'); + @json_stringify_append_char(str, ':'); + switch (key->type) { + case JSON_OBJECT: + case JSON_ARRAY: + @json_stringify_object_or_array(str, key->value); + break; + case JSON_STRING: + @json_stringify_append_char(str, '"'); + @json_stringify_append_str(str, key->value); + @json_stringify_append_char(str, '"'); + break; + case JSON_NUMBER: + @json_stringify_append_number(str, key->value); + break; + case JSON_BOOLEAN: + @json_stringify_append_str(str, @t(key->value, "true", "false")); + break; + case JSON_NULL: + @json_stringify_append_str(str, "null"); + break; + default: + PrintErr("@json_stringify_object: Invalid element type: %d", key->type); + while (1) { + Sleep(1); + }; + } + if (key->next) + @json_stringify_append_char(str, ','); + key = key->next; + } + @json_stringify_append_char(str, '}'); +} + +U0 @json_stringify_array(@json_stringify_string* str, @json_array* arr) +{ + @json_stringify_append_char(str, '['); + @json_item* item = arr->items; + while (item) { + switch (item->type) { + case JSON_OBJECT: + case JSON_ARRAY: + @json_stringify_object_or_array(str, item->value); + break; + case JSON_STRING: + @json_stringify_append_char(str, '"'); + @json_stringify_append_str(str, item->value); + @json_stringify_append_char(str, '"'); + break; + case JSON_NUMBER: + @json_stringify_append_number(str, item->value); + break; + case JSON_BOOLEAN: + @json_stringify_append_str(str, @t(item->value, "true", "false")); + break; + case JSON_NULL: + @json_stringify_append_str(str, "null"); + break; + default: + PrintErr("@json_stringify_array: Invalid element type: %d", item->type); + while (1) { + Sleep(1); + }; + } + if (item->next) + @json_stringify_append_char(str, ','); + item = item->next; + } + @json_stringify_append_char(str, ']'); +} + +U0 @json_stringify_object_or_array(@json_stringify_string* str, @json_element* el) +{ + while (el) { + switch (el->type) { + case JSON_OBJECT: + @json_stringify_object(str, el); + break; + case JSON_ARRAY: + @json_stringify_array(str, el); + break; + default: + PrintErr("@json_stringify_object_or_array: Invalid element type: %d", el->type); + while (1) { + Sleep(1); + }; + break; + } + el = el->next; + } +} + +U8* @json_stringify(@json_element* el, CTask* mem_task) +{ + if (!el || !mem_task) { + return NULL; + } + @json_stringify_string* str = CAlloc(sizeof(@json_stringify_string), mem_task); + str->mem_task = mem_task; + str->capacity = 256; + str->value = CAlloc(str->capacity * 2, str->mem_task); + @json_stringify_object_or_array(str, el); + return str->value; +} + +U64 @json_get(@json_object* obj, U8* key, Bool return_key = FALSE) +{ + if (!obj || !key) + return NULL; + if (!obj->keys || obj->type != JSON_OBJECT) + return NULL; + @json_key* iter_key = obj->keys; + while (iter_key) { + if (!StrICmp(iter_key->name, key)) + if (return_key) + return iter_key; + else + return iter_key->value; + iter_key = iter_key->next; + } + return NULL; +} + +U0 @json_set(@json_object* obj, U8* key, U64 value, I64 type = JSON_SAME) +{ + if (!obj || !key || !type) + return; + if (obj->type != JSON_OBJECT) + return; + @json_key* iter_key = obj->keys; + while (iter_key) { + if (!StrCmp(iter_key->name, key)) { + if (type != JSON_SAME) + iter_key->type = type; + iter_key->value = value; + return; + } + iter_key = iter_key->next; + } + @json_key* new_key = CAlloc(sizeof(@json_key), obj->mem_task); + new_key->sig = JSON_SIG; + new_key->name = StrNew(key, obj->mem_task); + new_key->type = type; + if (new_key->type == JSON_STRING) + new_key->value = StrNew(value, obj->mem_task); + else + new_key->value = value; + @json_insert_key(obj, new_key); +} + +U0 @json_unset(@json_object* obj, U8* key) +{ + if (!obj || !key) + return; + if (obj->type != JSON_OBJECT) + return; + @json_key* iter_key = obj->keys; + @json_key* prev_key = NULL; + @json_key* next_key = NULL; + while (iter_key) { + if (!StrCmp(iter_key->name, key)) { + prev_key = iter_key->prev; + next_key = iter_key->next; + if (prev_key) + prev_key->next = next_key; + if (next_key) + next_key->prev = prev_key; + return; + } + iter_key = iter_key->next; + } +} + +U64 @json_callable_object_get_wrapper_function(U8* key, Bool return_key = FALSE) +{ + return @json_get(JSON_WRAPPER_MAGIC_NUMBER, key, return_key); +} + +U0 @json_callable_object_set_wrapper_function(U8* key, U64 value, I64 type = JSON_SAME) +{ + @json_set(JSON_WRAPPER_MAGIC_NUMBER, key, value, type); +} + +U0 @json_callable_object_unset_wrapper_function(U8* key) +{ + @json_unset(JSON_WRAPPER_MAGIC_NUMBER, key); +} + +@json_callable_object* @json_create_callable_object(@json_object* obj) +{ + @json_callable_object* cobj = CAlloc(sizeof(@json_callable_object), obj->mem_task); + MemCpy(cobj, obj, sizeof(@json_object)); + + // Create a copy of function and patch Get + U64 a; + I64 code_size = MSize(&@json_callable_object_get_wrapper_function); + cobj->@ = CAlloc(code_size, obj->mem_task->code_heap); + MemCpy(cobj->@, &@json_callable_object_get_wrapper_function, code_size); + + a = cobj->@; + a += 0x13; + MemSetI64(a, cobj, 1); + + a = cobj->@; + a += 0x1c; + @patch_call_rel32(a, &@json_get); + + // Create a copy of function and patch Set + code_size = MSize(&@json_callable_object_set_wrapper_function); + cobj->set = CAlloc(code_size, obj->mem_task->code_heap); + MemCpy(cobj->set, &@json_callable_object_set_wrapper_function, code_size); + + a = cobj->set; + a += 0x1a; + MemSetI64(a, cobj, 1); + + a = cobj->set; + a += 0x23; + @patch_call_rel32(a, &@json_set); + + // Create a copy of function and patch Unset + code_size = MSize(&@json_callable_object_unset_wrapper_function); + cobj->unset = CAlloc(code_size, obj->mem_task->code_heap); + MemCpy(cobj->unset, &@json_callable_object_unset_wrapper_function, code_size); + + a = cobj->unset; + a += 0x0c; + MemSetI64(a, cobj, 1); + + a = cobj->unset; + a += 0x15; + @patch_call_rel32(a, &@json_unset); + + cobj->a = cobj->@; + cobj->o = cobj->@; + + return cobj; +} + +@json_callable_object* @json_create_object(CTask* mem_task) +{ + if (!mem_task) { + return NULL; + } + @json_object* obj = CAlloc(sizeof(@json_object), mem_task); + obj->mem_task = mem_task; + obj->sig = JSON_SIG; + obj->type = JSON_OBJECT; + return @json_create_callable_object(obj); +} + +U64 @json_array_index(@json_array* arr, I64 index, Bool return_item = FALSE) +{ + if (!arr) + return NULL; + if (arr->type != JSON_ARRAY) + return NULL; + if (index < 0) + return NULL; + if (!arr->length) + return NULL; + if (index > 0 && index > arr->length - 1) + return NULL; + @json_item* item = arr->items; + if (!item) + return NULL; + I64 i; + for (i = 0; i < index; i++) + item = item->next; + if (return_item) + return item; + else + return item->value; +} + +Bool @json_array_contains(@json_array* arr, U64 value, I64 type = NULL, Bool match_case = FALSE) +{ + if (!arr) + return FALSE; + if (arr->type != JSON_ARRAY) + return FALSE; + if (!arr->length) + return FALSE; + + if (!type) { + if (value > 0x1000) { + type = JSON_STRING; + } else { + type = JSON_BOOLEAN; + } + } + + I64 i; + @json_item* item = NULL; + for (i = 0; i < arr->length; i++) { + item = @json_array_index(arr, i, TRUE); + if (item->type == type) { + switch (type) { + case JSON_STRING: + if (match_case) { + if (!StrCmp(value, item->value)) { + return TRUE; + } + } else { + if (!StrICmp(value, item->value)) { + return TRUE; + } + } + break; + case JSON_BOOLEAN: + case JSON_NULL: + case JSON_NUMBER: + if (item->value == value) { + return TRUE; + } + break; + default: + break; + } + } + } + return FALSE; +} + +U0 @json_remove_item(@json_array* arr, I64 index) +{ + if (!arr) + return; + if (arr->type != JSON_ARRAY) + return; + if (index < 0) + return; + if (!arr->length) + return; + if (index > 0 && index > arr->length - 1) + return; + @json_item* item = arr->items; + if (!item) + return; + @json_item* prev_item = NULL; + @json_item* next_item = NULL; + I64 i; + for (i = 0; i < index; i++) + item = item->next; + if (!index) { + arr->items = item->next; + goto @json_remove_item_final; + } + prev_item = item->prev; + next_item = item->next; + if (arr->last_item == item) + arr->last_item = item->prev; + prev_item->next = next_item; + if (next_item) + next_item->prev = prev_item; + @json_remove_item_final : --arr->length; + if (!arr->length) + arr->last_item = NULL; +} + +U64 @json_callable_array_index_wrapper_function(I64 index, Bool return_item = FALSE) +{ + return @json_array_index(JSON_WRAPPER_MAGIC_NUMBER, index, return_item); +} + +U0 @json_callable_array_append_wrapper_function(U64 value, I64 type = NULL) +{ + @json_append_item(JSON_WRAPPER_MAGIC_NUMBER, value, type); +} + +Bool @json_callable_array_contains_wrapper_function(U64 value, I64 type = NULL, Bool match_case = FALSE) +{ + return @json_array_contains(JSON_WRAPPER_MAGIC_NUMBER, value, type, match_case); +} + +U0 @json_callable_array_insert_wrapper_function(I64 index, U64 value, I64 type = NULL) +{ + @json_insert_item(JSON_WRAPPER_MAGIC_NUMBER, index, value, type); +} + +U0 @json_callable_array_prepend_wrapper_function(U64 value, I64 type = NULL) +{ + @json_prepend_item(JSON_WRAPPER_MAGIC_NUMBER, value, type); +} + +U0 @json_callable_array_remove_wrapper_function(I64 index) +{ + @json_remove_item(JSON_WRAPPER_MAGIC_NUMBER, index); +} + +@json_callable_array* @json_create_callable_array(@json_array* arr) +{ + // Alloc callable object and copy instance + @json_callable_array* carr = CAlloc(sizeof(@json_callable_array), arr->mem_task); + MemCpy(carr, arr, sizeof(@json_array)); + + // Create a copy of function and patch Index + U64 a; + I64 code_size = MSize(&@json_callable_array_index_wrapper_function); + carr->@ = CAlloc(code_size, arr->mem_task->code_heap); + MemCpy(carr->@, &@json_callable_array_index_wrapper_function, code_size); + + a = carr->@; + a += 0x13; + MemSetI64(a, carr, 1); + + a = carr->@; + a += 0x1c; + @patch_call_rel32(a, &@json_array_index); + + carr->a = carr->@; + carr->o = carr->@; + + // Create a copy of function and patch Append + code_size = MSize(&@json_callable_array_append_wrapper_function); + carr->append = CAlloc(code_size, arr->mem_task->code_heap); + MemCpy(carr->append, &@json_callable_array_append_wrapper_function, code_size); + + a = carr->append; + a += 0x12; + MemSetI64(a, carr, 1); + + a = carr->append; + a += 0x1b; + @patch_call_rel32(a, &@json_append_item); + + // Create a copy of function and patch Contains + code_size = MSize(&@json_callable_array_contains_wrapper_function); + carr->contains = CAlloc(code_size, arr->mem_task->code_heap); + MemCpy(carr->contains, &@json_callable_array_contains_wrapper_function, code_size); + + a = carr->contains; + a += 0x1b; + MemSetI64(a, carr, 1); + + a = carr->contains; + a += 0x24; + @patch_call_rel32(a, &@json_array_contains); + + // Create a copy of function and patch Prepend + code_size = MSize(&@json_callable_array_prepend_wrapper_function); + carr->prepend = CAlloc(code_size, arr->mem_task->code_heap); + MemCpy(carr->prepend, &@json_callable_array_prepend_wrapper_function, code_size); + + a = carr->prepend; + a += 0x12; + MemSetI64(a, carr, 1); + + a = carr->prepend; + a += 0x1b; + @patch_call_rel32(a, &@json_prepend_item); + + // Create a copy of function and patch Insert + code_size = MSize(&@json_callable_array_insert_wrapper_function); + carr->insert = CAlloc(code_size, arr->mem_task->code_heap); + MemCpy(carr->insert, &@json_callable_array_insert_wrapper_function, code_size); + + a = carr->insert; + a += 0x1a; + MemSetI64(a, carr, 1); + + a = carr->insert; + a += 0x23; + @patch_call_rel32(a, &@json_insert_item); + + // Create a copy of function and patch Remove + code_size = MSize(&@json_callable_array_remove_wrapper_function); + carr->remove = CAlloc(code_size, arr->mem_task->code_heap); + MemCpy(carr->remove, &@json_callable_array_remove_wrapper_function, code_size); + + a = carr->remove; + a += 0x0c; + MemSetI64(a, carr, 1); + + a = carr->remove; + a += 0x15; + @patch_call_rel32(a, &@json_remove_item); + + return carr; +} + +@json_array* @json_create_array(CTask* mem_task) +{ + if (!mem_task) { + return NULL; + } + @json_array* arr = CAlloc(sizeof(@json_array), mem_task); + arr->mem_task = mem_task; + arr->sig = JSON_SIG; + arr->type = JSON_ARRAY; + return @json_create_callable_array(arr); +} + +U64 @json_parse_file(U8* path, CTask* mem_task) +{ + if (!path || !mem_task || !FileFind(path)) { + return NULL; + } + U64 res = NULL; + U8* json_string = FileRead(path); + if (json_string) { + res = @json_parse(json_string, mem_task); + } + return res; +} + +U0 @json_dump_to_file(U8* path, @json_element* el, CTask* mem_task) +{ + if (!path || !el || !mem_task) + return; + U8* json_string = @json_stringify(el, mem_task); + FileWrite(path, json_string, StrLen(json_string)); +} + +@json_array* @json_element_value_as_array(@json_element* el, CTask* mem_task, I64 key_or_item) +{ + if (!el || !mem_task || !key_or_item) { + return NULL; + } + switch (el->type) { + case JSON_ARRAY: + switch (key_or_item) { + case JSON_ELEMENT_IS_ITEM: + return el(@json_item*)->value; + case JSON_ELEMENT_IS_KEY: + return el(@json_key*)->value; + default: + return NULL; + } + break; + default: + break; + } + @json_array* arr = CAlloc(sizeof(@json_array), mem_task); + arr->mem_task = mem_task; + arr->sig = JSON_SIG; + arr->type = JSON_ARRAY; + switch (key_or_item) { + case JSON_ELEMENT_IS_ITEM: + @json_append_item(arr, el(@json_item*)->value, el->type); + break; + case JSON_ELEMENT_IS_KEY: + @json_append_item(arr, el(@json_key*)->value, el->type); + break; + default: + break; + } + return @json_create_callable_array(arr); +} + +@json_array* @json_item_value_as_array(@json_item* item, CTask* mem_task) +{ + return @json_element_value_as_array(item, mem_task, JSON_ELEMENT_IS_ITEM); +} + +@json_array* @json_key_value_as_array(@json_key* key, CTask* mem_task) +{ + return @json_element_value_as_array(key, mem_task, JSON_ELEMENT_IS_KEY); +} + +@json_element* @json_clone(@json_element* el, CTask* mem_task) +{ + if (!el || !mem_task) { + return NULL; + } + U8* tmp = @json_stringify(el, mem_task); + if (!tmp) { + return NULL; + } + return @json_parse(tmp, mem_task); +} + +class @json +{ + @json_element* (*Clone)(@json_element* el, CTask* mem_task); + @json_array* (*CreateArray)(CTask* mem_task); + @json_object* (*CreateObject)(CTask* mem_task); + U0 (*DumpToFile)(U8* path, @json_element* el, CTask* mem_task); + @json_array* (*KeyValueAsArray)(@json_key* key, CTask* mem_task); + @json_array* (*ItemValueAsArray)(@json_item* item, CTask* mem_task); + @json_element* (*Parse)(U8* str, CTask* mem_task); + U64 (*ParseFile)(U8* path, CTask* mem_task); + U8* (*Stringify)(@json_element* el, CTask* mem_task); +}; + +@json Json; +Json.Clone = &@json_clone; +Json.CreateArray = &@json_create_array; +Json.CreateObject = &@json_create_object; +Json.DumpToFile = &@json_dump_to_file; +Json.ItemValueAsArray = &@json_item_value_as_array; +Json.KeyValueAsArray = &@json_key_value_as_array; +Json.Parse = &@json_parse; +Json.ParseFile = &@json_parse_file; +Json.Stringify = &@json_stringify; + +"json "; diff --git a/System/Libraries/RawText.HC b/System/Libraries/RawText.HC new file mode 100644 index 0000000..5055c51 --- /dev/null +++ b/System/Libraries/RawText.HC @@ -0,0 +1,104 @@ +I64 @rawtext_output_port = NULL; + +U0 @rawtext_detect_qemu() +{ + CRAXRBCRCXRDX res; + CPUId(0x40000000, &res); + if (res.rbx == 'KVMK') + @rawtext_output_port = 0xe9; +} + +U0 @rawtext_detect_vbox() +{ + I64 res = PCIClassFind(0x088000, 0); + if (res >= 0) + @rawtext_output_port = 0x504; +} + +U0 @dbg_put_char(I64 ch) +{ + OutU8(@rawtext_output_port, ch); + if (!System.text_mode) + return; + Context2D* fb = Graphics2D.FrameBufferContext2D(); + text.raw_flags &= ~RWF_SHOW_DOLLAR; + if (ch > '~' && ch != 219) + ch = ' '; + I64 row, col; + if (!(text.raw_flags & RWF_SHOW_DOLLAR)) { + if (ch == '$$') { + if (text.raw_flags & RWF_IN_DOLLAR) { + text.raw_flags &= ~RWF_IN_DOLLAR; + if (!(text.raw_flags & RWF_LAST_DOLLAR)) { + text.raw_flags &= ~RWF_LAST_DOLLAR; + return; + } + } else { + text.raw_flags |= RWF_IN_DOLLAR | RWF_LAST_DOLLAR; + return; + } + } + text.raw_flags &= ~RWF_LAST_DOLLAR; + if (text.raw_flags & RWF_IN_DOLLAR) + return; + } + if (ch == '\t') { + @dbg_put_char(CH_SPACE); + while (text.raw_col & 7) + @dbg_put_char(CH_SPACE); + } else if (ch == CH_BACKSPACE) { + text.raw_col--; + @dbg_put_char(CH_SPACE); + text.raw_col--; + } else if (ch == '\n') { + @dbg_put_char(CH_SPACE); + while (text.raw_col % text.cols) + @dbg_put_char(CH_SPACE); + } else if (Bt(char_bmp_displayable, ch)) { + row = text.raw_col / text.cols % text.rows; + col = text.raw_col % text.cols; + if (text.raw_flags & RWF_SCROLL && text.raw_col && !row && !col) { + CopyRect2D(fb, 0, -16, fb); + Rect2D(fb, 0, Display.Height() - 16, Display.Width(), 16, 0x0); + text.raw_col -= text.cols; + row = text.rows - 1; + } + ConsolePrint2D(fb, col * 8, row * 16, , , "%c", ch); + text.raw_col++; + } +} + +Bool @kd_raw_putkey(I64 ch, I64) +{ + if (IsRaw) { + @dbg_put_char(ch); + return TRUE; + } else + return FALSE; +} + +Bool @kd_raw_puts(U8* st) +{ + I64 ch; + if (IsRaw) { + while (ch = *st++) + @dbg_put_char(ch); + return TRUE; + } else + return FALSE; +} + +U0 @rawdr_dummy(CTask*) { } + +CKeyDevEntry* tmp_kde = keydev.put_key_head; +while (tmp_kde->put_s != &KDRawPutS) + tmp_kde = tmp_kde->next; +tmp_kde->put_key = &@kd_raw_putkey; +tmp_kde->put_s = &@kd_raw_puts; + +Function.Patch(&RawDr, &@rawdr_dummy); + +@rawtext_detect_qemu; +@rawtext_detect_vbox; + +"rawtext "; \ No newline at end of file diff --git a/System/Libraries/Rsa.HC b/System/Libraries/Rsa.HC new file mode 100644 index 0000000..0b81bcf --- /dev/null +++ b/System/Libraries/Rsa.HC @@ -0,0 +1,46 @@ +Silent(1); // This is needed to suppress "Function should return val" warnings for wrappers to non-HolyC functions + +I64 @rsa_import(U8* der_bytes, I64 der_len, U64 key) +{ + U64 reg RDI rdi = der_bytes; + U64 reg RSI rsi = der_len; + U64 reg RDX rdx = key; + no_warn rdi, rsi, rdx; + asm { + MOV RAX, RSA_IMPORT + CALL RAX + } +} + +I64 @rsa_create_signature(U8* sig, I64* siglen, U8* hash, I64 hashlen, U64 key) +{ + U64 reg RDI rdi = sig; + U64 reg RSI rsi = siglen; + U64 reg RDX rdx = hash; + U64 reg RCX rcx = hashlen; + U64 reg R8 r8 = key; + no_warn rdi, rsi, rdx, rcx, r8; + asm { + MOV RAX, RSA_CREATE_SIGNATURE + CALL RAX + } +} + +I64 @rsa_verify_signature(U8* sig, I64 siglen, U8* hash, I64 hashlen, I32* stat, U64 key) +{ + U64 reg RDI rdi = sig; + U64 reg RSI rsi = siglen; + U64 reg RDX rdx = hash; + U64 reg RCX rcx = hashlen; + U64 reg R8 r8 = stat; + U64 reg R9 r9 = key; + no_warn rdi, rsi, rdx, rcx, r8, r9; + asm { + MOV RAX, RSA_VERIFY_SIGNATURE + CALL RAX + } +} + +Silent(0); + +"rsa "; diff --git a/System/Libraries/Session.HC b/System/Libraries/Session.HC new file mode 100644 index 0000000..ecf3191 --- /dev/null +++ b/System/Libraries/Session.HC @@ -0,0 +1,8 @@ +class @session +{ + U8 home[4096]; + U8 hostname[256]; + @user user; +}; + +"session "; \ No newline at end of file diff --git a/System/Libraries/Sha256.HC b/System/Libraries/Sha256.HC new file mode 100644 index 0000000..71e80b2 --- /dev/null +++ b/System/Libraries/Sha256.HC @@ -0,0 +1,235 @@ +#define CHUNK_SIZE 64 +#define TOTAL_LEN_LEN 8 + +/* + * ABOUT bool: this file does not use bool in order to be as pre-C99 compatible + * as possible. + */ + +/* + * Comments from pseudo-code at https://en.wikipedia.org/wiki/SHA-2 are + * reproduced here. When useful for clarification, portions of the pseudo-code + * are reproduced here too. + */ + +/* + * Initialize array of round constants: + * (first 32 bits of the fractional parts of the cube roots of the first 64 + * primes 2..311): + */ +U32 k[64] = { + 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1, + 0x923f82a4, 0xab1c5ed5, 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, + 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, 0xe49b69c1, 0xefbe4786, + 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, + 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147, + 0x06ca6351, 0x14292967, 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, + 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, 0xa2bfe8a1, 0xa81a664b, + 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070, + 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, + 0x5b9cca4f, 0x682e6ff3, 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, + 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2 +}; + +U32 _h[8] = { 0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a, + 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19 }; + +class buffer_state { + U8* p; + I32 len; + I32 total_len; + I32 single_one_delivered; /* bool */ + I32 total_len_delivered; /* bool */ +}; + +U32 right_rot(U32 value, U32 count) +{ + /* + * Defined behaviour in standard C for all count where 0 < count < 32, + * which is what we need here. + */ + return value >> count | value << (32 - count); +} + +U0 init_buf_state(buffer_state* state, U8* input, I32 len) +{ + state->p = input; + state->len = len; + state->total_len = len; + state->single_one_delivered = 0; + state->total_len_delivered = 0; +} + +/* Return value: bool */ +I32 calc_chunk(U8* chunk, buffer_state* state) +{ + I32 space_in_chunk; + + if (state->total_len_delivered) { + return 0; + } + + if (state->len >= CHUNK_SIZE) { + MemCpy(chunk, state->p, CHUNK_SIZE); + state->p += CHUNK_SIZE; + state->len -= CHUNK_SIZE; + return 1; + } + + MemCpy(chunk, state->p, state->len); + chunk += state->len; + space_in_chunk = CHUNK_SIZE - state->len; + state->p += state->len; + state->len = 0; + + /* If we are here, space_in_chunk is one at minimum. */ + if (!state->single_one_delivered) { + *chunk++ = 0x80; + space_in_chunk -= 1; + state->single_one_delivered = 1; + } + + /* + * Now: + * - either there is enough space left for the total length, and we can + * conclude, + * - or there is too little space left, and we have to pad the rest of this + * chunk with zeroes. In the latter case, we will conclude at the next + * invokation of this function. + */ + if (space_in_chunk >= TOTAL_LEN_LEN) { + I32 left = space_in_chunk - TOTAL_LEN_LEN; + I32 len = state->total_len; + I32 i; + MemSet(chunk, 0x00, left); + chunk += left; + + /* Storing of len * 8 as a big endian 64-bit without overflow. */ + chunk[7] = (len << 3); + len >>= 5; + for (i = 6; i >= 0; i--) { + chunk[i] = len; + len >>= 8; + } + state->total_len_delivered = 1; + } else { + MemSet(chunk, 0x00, space_in_chunk); + } + + return 1; +} + +/* + * Limitations: + * - Since input is a pointer in RAM, the data to hash should be in RAM, which + * could be a problem for large data sizes. + * - SHA algorithms theoretically operate on bit strings. However, this + * implementation has no support for bit string lengths that are not multiples + * of eight, and it really operates on arrays of bytes. In particular, the len + * parameter is a number of bytes. + */ +U0 calc_sha_256(U8* hash, U8* input, I32 len) +{ + /* + * Note 1: All integers (expect indexes) are 32-bit U32 integers and addition + * is calculated modulo 2^32. Note 2: For each round, there is one round + * constant k[i] and one entry in the message schedule array w[i], 0 = i = 63 + * Note 3: The compression function uses 8 working variables, a through h + * Note 4: Big-endian convention is used when expressing the constants in this + * pseudocode, and when parsing message block data from bytes to words, for + * example, the first word of the input message "abc" after padding is + * 0x61626380 + */ + + /* + * Initialize hash values: + * (first 32 bits of the fractional parts of the square roots of the first 8 + * primes 2..19): + */ + + U32 h[8]; + U32 i, j; + MemCpy(h, _h, sizeof(U32) * 8); + + /* 512-bit chunks is what we will operate on. */ + U8 chunk[64]; + + buffer_state state; + + init_buf_state(&state, input, len); + + while (calc_chunk(chunk, &state)) { + U32 ah[8]; + + U8* p = chunk; + + /* Initialize working variables to current hash value: */ + for (i = 0; i < 8; i++) + ah[i] = h[i]; + + /* Compression function main loop: */ + for (i = 0; i < 4; i++) { + /* + * The w-array is really w[64], but since we only need + * 16 of them at a time, we save stack by calculating + * 16 at a time. + * + * This optimization was not there initially and the + * rest of the comments about w[64] are kept in their + * initial state. + */ + + /* + * create a 64-entry message schedule array w[0..63] of 32-bit words + * (The initial values in w[0..63] don't matter, so many implementations + * zero them here) copy chunk into first 16 words w[0..15] of the message + * schedule array + */ + U32 w[16]; + + U32 s0, s1; + + for (j = 0; j < 16; j++) { + if (i == 0) { + w[j] = p[0] << 24 | p[1] << 16 | p[2] << 8 | p[3]; + p += 4; + } else { + /* Extend the first 16 words into the remaining 48 words w[16..63] of + * the message schedule array: */ + s0 = right_rot(w[(j + 1) & 0xf], 7) ^ right_rot(w[(j + 1) & 0xf], 18) ^ (w[(j + 1) & 0xf] >> 3); + s1 = right_rot(w[(j + 14) & 0xf], 17) ^ right_rot(w[(j + 14) & 0xf], 19) ^ (w[(j + 14) & 0xf] >> 10); + w[j] = w[j] + s0 + w[(j + 9) & 0xf] + s1; + } + s1 = right_rot(ah[4], 6) ^ right_rot(ah[4], 11) ^ right_rot(ah[4], 25); + U32 ch = (ah[4] & ah[5]) ^ (~ah[4] & ah[6]); + U32 temp1 = ah[7] + s1 + ch + k[i << 4 | j] + w[j]; + s0 = right_rot(ah[0], 2) ^ right_rot(ah[0], 13) ^ right_rot(ah[0], 22); + U32 maj = (ah[0] & ah[1]) ^ (ah[0] & ah[2]) ^ (ah[1] & ah[2]); + U32 temp2 = s0 + maj; + + ah[7] = ah[6]; + ah[6] = ah[5]; + ah[5] = ah[4]; + ah[4] = ah[3] + temp1; + ah[3] = ah[2]; + ah[2] = ah[1]; + ah[1] = ah[0]; + ah[0] = temp1 + temp2; + } + } + + /* Add the compressed chunk to the current hash value: */ + for (i = 0; i < 8; i++) + h[i] += ah[i]; + } + + /* Produce the final hash value (big-endian): */ + for (i = 0, j = 0; i < 8; i++) { + hash[j++] = (h[i] >> 24); + hash[j++] = (h[i] >> 16); + hash[j++] = (h[i] >> 8); + hash[j++] = h[i]; + } +} + +"sha256 "; diff --git a/System/Libraries/Shell.HC b/System/Libraries/Shell.HC new file mode 100644 index 0000000..7c5b9c1 --- /dev/null +++ b/System/Libraries/Shell.HC @@ -0,0 +1,52 @@ +#define SHELL_HISTORY_LIMIT 1025 +#define SHELL_INPUT_FIFO_SIZE 65536 + +class @shell_env_var +{ + @shell_env_var* prev; + @shell_env_var* next; + U8 key[256]; + U8 value[1024]; +}; + +class @shell_autocomplete +{ + I64 depth; + I64 length[8]; + U8*** entries; +}; + +class @shell_history +{ + I64 index; + I64 limit; + I64 pos; + U8** entries; +}; + +class @shell_readline +{ + @shell_autocomplete autocomplete; + @shell_history history; +}; + +class @shell +{ + CFifoU8* input; + CFifoU8* output; + CTask* task; + Bool break; + Bool exit; + I64 answer; + @shell_env_var* env; + @shell_history history; + @shell_readline readline; + @session* session; + U8 cwd[4096]; + U8 PS1[512]; + U8 PS2[512]; + U8 PS3[512]; + U8 PS4[512]; +}; + +"shell "; \ No newline at end of file diff --git a/System/Libraries/Stdio.HC b/System/Libraries/Stdio.HC new file mode 100644 index 0000000..877719c --- /dev/null +++ b/System/Libraries/Stdio.HC @@ -0,0 +1,155 @@ +class @stdio +{ + U0 (*ReadLine)(@shell* sh, U8* str); + U0 (*WriteLine)(@shell* sh, U8* fmt, ...); +}; + +U0 @stdio_write_line(@shell* sh, U8* fmt, ...) +{ + if (!sh) + return; + if (!fmt || !sh->output) + return; + U8* buf; + if (argc) { + buf = StrPrintJoin(NULL, fmt, argc, argv); + } else { + buf = StrNew(fmt, erythros_mem_task); + } + I64 i; + for (i = 0; i < StrLen(buf); i++) + FifoU8Ins(sh->output, buf[i]); + Free(buf); +} + +I64 @stdio_handle_control_chars(@shell* sh) +{ + if (!FifoU8Cnt(sh->input)) + return 0; + U8 char; + FifoU8Rem(sh->input, &char); + switch (char) { + case '[': + if (!FifoU8Cnt(sh->input)) + return 0; + FifoU8Rem(sh->input, &char); + switch (char) { + case 'A': + return SC_CURSOR_UP; + break; + case 'B': + return SC_CURSOR_DOWN; + break; + case 'D': + return SC_CURSOR_LEFT; + break; + case 'C': + return SC_CURSOR_RIGHT; + break; + default: + return 0; + break; + } + break; + default: + return 0; + break; + } +} + +U0 @stdio_read_line_history_back(@shell* sh, I64 pos) +{ + if (sh->history.index < 0) + sh->history.index = 0; + while (pos > 0) { + FifoU8Ins(sh->input, '\x8'); + pos--; + } + U8* char = sh->history.entries[sh->history.index]; + while (*char) + FifoU8Ins(sh->input, *char ++); + if (sh->history.index > -1) { + sh->history.index--; + } +} + +U0 @stdio_read_line_history_fwd(@shell* sh, I64 pos) +{ + if (sh->history.index < sh->history.pos) { + sh->history.index++; + } + if (sh->history.index > sh->history.pos) + sh->history.index = sh->history.pos; + while (pos > 0) { + FifoU8Ins(sh->input, '\x8'); + pos--; + } + U8* char = sh->history.entries[sh->history.index]; + while (*char) + FifoU8Ins(sh->input, *char ++); +} + +U0 @stdio_read_line(@shell* sh, U8* str) +{ + U8 char = NULL; + U8 line[4096]; + I64 pos = 0; + if (!str || !sh) + return; + sh->history.index = sh->history.pos - 1; + while (char != '\x3' && char != '\n') { + while (FifoU8Cnt(sh->input)) { + FifoU8Rem(sh->input, &char); + switch (char) { + case 3: + @stdio_write_line(sh, "^C"); + break; + case 8: + if (pos > 0) { + line[StrLen(line) - 1] = NULL; + FifoU8Ins(sh->output, '\x8'); + pos--; + } else + FifoU8Ins(sh->output, '\x7'); + break; + case 13: + break; + case 27: + switch (@stdio_handle_control_chars(sh)) { + case SC_CURSOR_UP: + @stdio_read_line_history_back(sh, pos); + break; + case SC_CURSOR_DOWN: + @stdio_read_line_history_fwd(sh, pos); + break; + default: + break; + } + break; + case 32...127: + line[pos] = char; + FifoU8Ins(sh->output, char); + pos++; + break; + }; + } + Sleep(1); + } + line[pos] = NULL; + switch (char) { + case '\x3': + StrCpy(str, ""); + break; + case '\n': + StrCpy(str, &line); + break; + }; + FifoU8Ins(sh->output, '\n'); +} + +@stdio Stdio; + +Stdio.ReadLine = &@stdio_read_line; +Stdio.WriteLine = &@stdio_write_line; + +"stdio "; \ No newline at end of file diff --git a/System/Libraries/String.HC b/System/Libraries/String.HC new file mode 100644 index 0000000..bca0e71 --- /dev/null +++ b/System/Libraries/String.HC @@ -0,0 +1,171 @@ +#define TRIM_BOTH 0 +#define TRIM_LEFT 1 +#define TRIM_RIGHT 2 + +U0 @string_append(U8* dst, U8* fmt, ...) +{ + U8* buf; + if (argc) { + buf = StrPrintJoin(NULL, fmt, argc, argv); + } else { + buf = StrNew(fmt, erythros_mem_task); + } + U8* src = buf; + StrCpy(dst + StrLen(dst), src); + Free(buf); +} + +Bool @string_is_number(U8* s) +{ + while (*s) { + switch (*s) { + case '-': + case '.': + case '0' ... '9': + break; + default: + return FALSE; + break; + } + s++; + } + return TRUE; +} + +Bool @string_begins_with(U8* fragment, U8* str) +{ + if (!fragment || !str) + return FALSE; + if (StrLen(fragment) > StrLen(str)) + return FALSE; + return !MemCmp(fragment, str, StrLen(fragment)); +} + +Bool @string_ends_with(U8* fragment, U8* str) +{ + if (!fragment || !str) + return FALSE; + if (StrLen(fragment) > StrLen(str)) + return FALSE; + return !MemCmp(fragment, str + StrLen(str) - StrLen(fragment), StrLen(fragment)); +} + +U8* @string_replace(U8* s, U8* oldW, U8* newW) +{ + if (!StrFind(oldW, s)) { + return StrNew(s, erythros_mem_task); + } + U8* result; + I64 i, cnt = 0; + I64 newWlen = StrLen(newW); + I64 oldWlen = StrLen(oldW); + for (i = 0; s[i] != NULL; i++) { + if (StrFind(oldW, &s[i]) == &s[i]) { + cnt++; + + i += oldWlen - 1; + } + } + result = MAlloc(i + cnt * (newWlen - oldWlen) + 1, erythros_mem_task); + i = 0; + while (*s) { + if (StrFind(oldW, s) == s) { + StrCpy(&result[i], newW); + i += newWlen; + s += oldWlen; + } else + result[i++] = *s++; + } + result[i] = NULL; + return result; +} + +U8** @string_split(U8* s, U8 ch = '\n', I64* cnt) +{ + U8 check_buf[4]; + StrPrint(check_buf, "%c", ch); + if (!StrFind(check_buf, s)) { + U8** same_arr = CAlloc(sizeof(U8*) * 1, erythros_mem_task); + same_arr[0] = s; + *cnt = 1; + return same_arr; + } + U8* p = s; + cnt[0] = 0; + while (*p) { + if (*p == ch) + cnt[0]++; + p++; + } + if (!(cnt[0])) + return NULL; + cnt[0]++; + I64 i = -1; + U8** arr = CAlloc(sizeof(U8*) * cnt[0], erythros_mem_task); + p = s; + while (*p) { + if (*p == ch || i < 0) { + i++; + arr[i] = p; + if (*p == ch) { + arr[i]++; + *p = NULL; + } + } + p++; + } + return arr; +} + +Bool @string_trim_ch(U8 s_ch, U8 trim_ch) +{ + if (!s_ch) { + return FALSE; + } + if (!trim_ch) { + return (s_ch == ' ' || s_ch == '\r' || s_ch == '\n' || s_ch == '\t'); + } else { + return (s_ch == trim_ch); + } +} + +U0 @string_trim(U8* s, U8 ch = NULL, I64 mode = TRIM_BOTH) +{ + Bool trim_ch = @string_trim_ch(*s, ch); + if (mode == TRIM_BOTH || mode == TRIM_LEFT) { + while (trim_ch) { + StrCpy(s, s + 1); + trim_ch = @string_trim_ch(*s, ch); + } + } + trim_ch = @string_trim_ch(s[StrLen(s) - 1], ch); + if (mode == TRIM_BOTH || mode == TRIM_RIGHT) { + while (trim_ch) { + s[StrLen(s) - 1] = NULL; + trim_ch = @string_trim_ch(s[StrLen(s) - 1], ch); + } + } +} + +class @string +{ + U0(*Append) + (U8 * dst, U8 * fmt, ...); + Bool (*BeginsWith)(U8* fragment, U8* str); + Bool (*EndsWith)(U8* fragment, U8* str); + Bool (*IsNumber)(U8* s); + U8* (*Replace)(U8* s, U8* oldW, U8* newW); + U8** (*Split)(U8* s, U8 ch = '\n', I64 * cnt); + U0 (*Trim)(U8* s, U8 ch = NULL, I64 mode = TRIM_BOTH); +}; + +@string String; +String.Append = &@string_append; +String.BeginsWith = &@string_begins_with; +String.EndsWith = &@string_ends_with; +String.IsNumber = &@string_is_number; +String.Replace = &@string_replace; +String.Split = &@string_split; +String.Trim = &@string_trim; + +"string "; diff --git a/System/Libraries/System.HC b/System/Libraries/System.HC new file mode 100644 index 0000000..3ad4085 --- /dev/null +++ b/System/Libraries/System.HC @@ -0,0 +1,104 @@ +class @system +{ + Bool text_mode; + CFifoI64* log_fifo; + U8* build_info; + U8* (*BuildInfo)(); + U0 (*Init)(); + U0 (*Log)(CTask* task, U8* fmt, ...); + U0 (*PowerOff)(); +}; + +@system System; + +U0 @system_lex_warn(CCmpCtrl* cc, + U8* str = NULL) +{ // Print warn msg, then, LexPutPos(). + if (!MemCmp(str, "Assign U0 ", 10)) + return; // suppress "Assign U0 " warnings + if (str) + PrintWarn(str); + if (cc->htc.fun) { + "in fun '%s'.\n", cc->htc.fun->str; + if (IsRaw) + "%s\n", cc->htc.fun->src_link; + else { + "$LK,\"%s\"$\n", cc->htc.fun->src_link; + AdamErr("%s\n", cc->htc.fun->src_link); + } + } else + LexPutPos(cc); + cc->warning_cnt++; +} + +U0 @system_print_warn(U8* fmt, ...) +{ // Print "Warn:" and msg in blinking red. + if (!MemCmp(fmt, "Unused var", 10)) + return; // suppress "Unused var" warnings + if (!MemCmp(fmt, "Using 64-bit reg var.", 21)) + return; // suppress "Using 64-bit reg var." warnings + U8* buf = StrPrintJoin(NULL, fmt, argc, argv); + GetOutOfDollar; + "%,p %,p %,p %,p " ST_WARN_ST "%s", Caller, Caller(2), Caller(3), Caller(4), + buf; + Free(buf); +} + +@patch_jmp_rel32(&LexWarn, &@system_lex_warn); +@patch_jmp_rel32(&PrintWarn, &@system_print_warn); + +U8* @system_build_info() { return System.build_info; } + +U0 @system_log(CTask* task, U8* fmt, ...) +{ + U8* buf = StrPrintJoin(NULL, fmt, argc, argv); + U8* str = buf; + U32 color; + MemCpyU32(&color, &task->task_name, 1); + if (!color) { + color = RandU32 * 1048576; + MemCpyU32(&task->pad, &color, 1); + } + U8* log_msg = MAlloc(1024); + StrPrint(log_msg, "[\x1b[38;2;%d;%d;%dm%16s\x1b[0m] %s\n", + color.u8[3] << 5 & 0xFF, color.u8[2] << 4 & 0xFF, + color.u8[1] << 3 & 0xFF, Fs->task_name, buf); + FifoI64Ins(System.log_fifo, log_msg); + Free(buf); +} + +U0 @system_log_task() +{ + I64 log_msg; + while (1) { + while (FifoI64Cnt(System.log_fifo)) { + FifoI64Rem(System.log_fifo, &log_msg); + "%s", log_msg; + Free(log_msg); + } + Sleep(1); + } +} + +U0 @system_power_off() +{ + OutU16(0x4004, 0x3400); + OutU16(0x0604, 0x2000); + OutU16(0xB004, 0x2000); +} + +U0 @system_init() +{ + System.build_info = FileRead("build_info.TXT"); + System.log_fifo = FifoI64New(1024); + Spawn(&@system_log_task, , , T(mp_cnt, 1, 0)); +} + +System.BuildInfo = &@system_build_info; +System.Init = &@system_init; +System.Log = &@system_log; +System.PowerOff = &@system_power_off; + +System.Init(); + +"system "; \ No newline at end of file diff --git a/System/Libraries/Theme.HC b/System/Libraries/Theme.HC new file mode 100644 index 0000000..6e194f1 --- /dev/null +++ b/System/Libraries/Theme.HC @@ -0,0 +1,49 @@ +class @theme_colors +{ + U32 active_border; + U32 hilight; +}; + +class @theme_window +{ + I64 min_width; + I64 min_height; +}; + +class @theme_bitmap_fonts +{ + BitmapFont* menu; + BitmapFont* monospace; + BitmapFont* sans; +} + +class @theme_pointers +{ + Context2D* pointer; + Context2D* pen; + Context2D* move; + Context2D* link; + AnimationContext2D* wait; + Context2D* horz; + Context2D* vert; + Context2D* text; + Context2D* cross; + Context2D* dgn1; + Context2D* dgn2; + Context2D* help; + Context2D* alternate; + Context2D* unavailable; +}; + +class @theme +{ + U8* path; + @theme_colors color; + @theme_bitmap_fonts font; + @theme_pointers pointer; + @theme_window window; + Context2D* wallpaper; + U0 (*window_repaint)(Window* win, I64 type); +}; + +"theme "; \ No newline at end of file diff --git a/System/Libraries/Tlse.HC b/System/Libraries/Tlse.HC new file mode 100644 index 0000000..62fb3d3 --- /dev/null +++ b/System/Libraries/Tlse.HC @@ -0,0 +1,115 @@ +#define TLS_V12 0x0303 + +Silent(1); // This is needed to suppress "Function should return val" warnings for wrappers to non-HolyC functions + +U64 @tls_create_context(U8 is_server, U16 version) +{ + U64 reg RDI rdi = is_server; + U64 reg RSI rsi = version; + no_warn rdi, rsi; + asm { + MOV RAX, TLS_CREATE_CONTEXT + CALL RAX + } +} + +I32 @tls_sni_set(U64 context, U8* sni) +{ + U64 reg RDI rdi = context; + U64 reg RSI rsi = sni; + no_warn rdi, rsi; + asm { + MOV RAX, TLS_SNI_SET + CALL RAX + } +} + +I32 @tls_client_connect(U64 context) +{ + U64 reg RDI rdi = context; + no_warn rdi; + asm { + MOV RAX, TLS_CLIENT_CONNECT + CALL RAX + } +} + +U8* @tls_get_write_buffer(U64 context, U32* outlen) +{ + U64 reg RDI rdi = context; + U64 reg RSI rsi = outlen; + no_warn rdi, rsi; + asm { + MOV RAX, TLS_GET_WRITE_BUFFER + CALL RAX + } +} + +U0 @tls_buffer_clear(U64 context) +{ + U64 reg RDI rdi = context; + no_warn rdi; + asm { + MOV RAX, TLS_BUFFER_CLEAR + CALL RAX + } +} + +I32 @tls_connection_status(U64 context) +{ + U64 reg RDI rdi = context; + no_warn rdi; + asm { + MOV RAX, TLS_CONNECTION_STATUS + CALL RAX + } +} + +U0 @tls_consume_stream(U64 context, U8* buf, I32 buf_len, U64 certificate_verify) +{ + U64 reg RDI rdi = context; + U64 reg RSI rsi = buf; + U64 reg RDX rdx = buf_len; + U64 reg RCX rcx = certificate_verify; + no_warn rdi, rsi, rdx, rcx; + asm { + MOV RAX, TLS_CONSUME_STREAM + CALL RAX + } +} + +I32 @tls_read(U64 context, U8* buf, U32 size) +{ + U64 reg RDI rdi = context; + U64 reg RSI rsi = buf; + U64 reg RDX rdx = size; + no_warn rdi, rsi, rdx; + asm { + MOV RAX, TLS_READ + CALL RAX + } +} + +I32 @tls_write(U64 context, U8* data, U32 len) +{ + U64 reg RDI rdi = context; + U64 reg RSI rsi = data; + U64 reg RDX rdx = len; + no_warn rdi, rsi, rdx; + asm { + MOV RAX, TLS_WRITE + CALL RAX + } +} + +I32 @tls_established(U64 context) +{ + U64 reg RDI rdi = context; + no_warn rdi; + asm { + MOV RAX, TLS_ESTABLISHED + CALL RAX + } +} + +Silent(0); diff --git a/System/Libraries/User.HC b/System/Libraries/User.HC new file mode 100644 index 0000000..6a44a69 --- /dev/null +++ b/System/Libraries/User.HC @@ -0,0 +1,10 @@ +class @user +{ + I64 uid; + U8 name[256]; + U8 fullname[256]; + U8 passwd[256]; + U8 groups[256]; +} + +"user "; \ No newline at end of file diff --git a/System/Libraries/Widget.HC b/System/Libraries/Widget.HC new file mode 100644 index 0000000..a516505 --- /dev/null +++ b/System/Libraries/Widget.HC @@ -0,0 +1,834 @@ +#define TextInputWidget BitmapFontTextInputWidget +#define TextLabelWidget BitmapFontTextLabelWidget + +#define WIDGET_TYPE_NULL 0 +#define WIDGET_TYPE_BUTTON 1 +#define WIDGET_TYPE_CHECKBOX 2 +#define WIDGET_TYPE_RADIO 3 +#define WIDGET_TYPE_INPUT 4 +#define WIDGET_TYPE_LABEL 5 +#define WIDGET_TYPE_CONTEXT2D 6 +#define WIDGET_TYPE_TTF_INPUT 7 +#define WIDGET_TYPE_TTF_LABEL 8 +#define WIDGET_TYPE_HORZ_SLIDER 9 +#define WIDGET_TYPE_VERT_SLIDER 10 +#define WIDGET_TYPE_TERMINAL 11 +#define WIDGET_TYPE_HORZ_SCROLLBAR 12 +#define WIDGET_TYPE_VERT_SCROLLBAR 13 +#define WIDGET_TYPE_MENU_ITEM 14 +#define WIDGET_TYPE_LISTVIEW 15 + +#define TERMINAL_MAX_COLS 1920 / 8 + +#define TERMINAL_STATE_OUTPUT 0 +#define TERMINAL_STATE_CONSUME_BEGIN 1 +#define TERMINAL_STATE_CONSUME_CTRL_SEQ 2 +#define TERMINAL_STATE_CONSUME_OS_CMD 3 +#define TERMINAL_STATE_CONSUME_END 99 + +asm { +TERMINAL_COLOR_TABLE:: +DU32 0xff000000, 0xff800000, 0xff008000, 0xff808000, 0xff000080, 0xff800080, 0xff008080, 0xffc0c0c0, 0xff808080, 0xffff0000, 0xff00ff00, 0xffffff00, 0xff0000ff, 0xffff00ff, 0xff00ffff, 0xffffffff, 0xff000000, 0xff00005f, 0xff000087, 0xff0000af, 0xff0000d7, 0xff0000ff, 0xff005f00, 0xff005f5f, 0xff005f87, 0xff005faf, 0xff005fd7, 0xff005fff, 0xff008700, 0xff00875f, 0xff008787, 0xff0087af, 0xff0087d7, 0xff0087ff, 0xff00af00, 0xff00af5f, 0xff00af87, 0xff00afaf, 0xff00afd7, 0xff00afff, 0xff00d700, 0xff00d75f, 0xff00d787, 0xff00d7af, 0xff00d7d7, 0xff00d7ff, 0xff00ff00, 0xff00ff5f, 0xff00ff87, 0xff00ffaf, 0xff00ffd7, 0xff00ffff, 0xff5f0000, 0xff5f005f, 0xff5f0087, 0xff5f00af, 0xff5f00d7, 0xff5f00ff, 0xff5f5f00, 0xff5f5f5f, 0xff5f5f87, 0xff5f5faf, 0xff5f5fd7, 0xff5f5fff, 0xff5f8700, 0xff5f875f, 0xff5f8787, 0xff5f87af, 0xff5f87d7, 0xff5f87ff, 0xff5faf00, 0xff5faf5f, 0xff5faf87, 0xff5fafaf, 0xff5fafd7, 0xff5fafff, 0xff5fd700, 0xff5fd75f, 0xff5fd787, 0xff5fd7af, 0xff5fd7d7, 0xff5fd7ff, 0xff5fff00, 0xff5fff5f, 0xff5fff87, 0xff5fffaf, 0xff5fffd7, 0xff5fffff, 0xff870000, 0xff87005f, 0xff870087, 0xff8700af, 0xff8700d7, 0xff8700ff, 0xff875f00, 0xff875f5f, 0xff875f87, 0xff875faf, 0xff875fd7, 0xff875fff, 0xff878700, 0xff87875f, 0xff878787, 0xff8787af, 0xff8787d7, 0xff8787ff, 0xff87af00, 0xff87af5f, 0xff87af87, 0xff87afaf, 0xff87afd7, 0xff87afff, 0xff87d700, 0xff87d75f, 0xff87d787, 0xff87d7af, 0xff87d7d7, 0xff87d7ff, 0xff87ff00, 0xff87ff5f, 0xff87ff87, 0xff87ffaf, 0xff87ffd7, 0xff87ffff, 0xffaf0000, 0xffaf005f, 0xffaf0087, 0xffaf00af, 0xffaf00d7, 0xffaf00ff, 0xffaf5f00, 0xffaf5f5f, 0xffaf5f87, 0xffaf5faf, 0xffaf5fd7, 0xffaf5fff, 0xffaf8700, 0xffaf875f, 0xffaf8787, 0xffaf87af, 0xffaf87d7, 0xffaf87ff, 0xffafaf00, 0xffafaf5f, 0xffafaf87, 0xffafafaf, 0xffafafd7, 0xffafafff, 0xffafd700, 0xffafd75f, 0xffafd787, 0xffafd7af, 0xffafd7d7, 0xffafd7ff, 0xffafff00, 0xffafff5f, 0xffafff87, 0xffafffaf, 0xffafffd7, 0xffafffff, 0xffd70000, 0xffd7005f, 0xffd70087, 0xffd700af, 0xffd700d7, 0xffd700ff, 0xffd75f00, 0xffd75f5f, 0xffd75f87, 0xffd75faf, 0xffd75fd7, 0xffd75fff, 0xffd78700, 0xffd7875f, 0xffd78787, 0xffd787af, 0xffd787d7, 0xffd787ff, 0xffd7af00, 0xffd7af5f, 0xffd7af87, 0xffd7afaf, 0xffd7afd7, 0xffd7afff, 0xffd7d700, 0xffd7d75f, 0xffd7d787, 0xffd7d7af, 0xffd7d7d7, 0xffd7d7ff, 0xffd7ff00, 0xffd7ff5f, 0xffd7ff87, 0xffd7ffaf, 0xffd7ffd7, 0xffd7ffff, 0xffff0000, 0xffff005f, 0xffff0087, 0xffff00af, 0xffff00d7, 0xffff00ff, 0xffff5f00, 0xffff5f5f, 0xffff5f87, 0xffff5faf, 0xffff5fd7, 0xffff5fff, 0xffff8700, 0xffff875f, 0xffff8787, 0xffff87af, 0xffff87d7, 0xffff87ff, 0xffffaf00, 0xffffaf5f, 0xffffaf87, 0xffffafaf, 0xffffafd7, 0xffffafff, 0xffffd700, 0xffffd75f, 0xffffd787, 0xffffd7af, 0xffffd7d7, 0xffffd7ff, 0xffffff00, 0xffffff5f, 0xffffff87, 0xffffffaf, 0xffffffd7, 0xffffffff, 0xff080808, 0xff121212, 0xff1c1c1c, 0xff262626, 0xff303030, 0xff3a3a3a, 0xff444444, 0xff4e4e4e, 0xff585858, 0xff626262, 0xff6c6c6c, 0xff767676, 0xff808080, 0xff8a8a8a, 0xff949494, 0xff9e9e9e, 0xffa8a8a8, 0xffb2b2b2, 0xffbcbcbc, 0xffc6c6c6, 0xffd0d0d0, 0xffdadada, 0xffe4e4e4, 0xffeeeeee; +} + +U8 widget_self_set1[0x1F] = { 0x55, 0x48, 0x8B, 0xEC, 0x56, 0x48, 0x8B, 0x75, + 0x10, 0x56, 0x48, 0xBB, 0x0, 0x0, 0x0, 0x0, + 0x0, 0x0, 0x0, 0x0, 0x53, 0xE8, 0x0, 0x0, + 0x0, 0x0, 0x5E, 0x5D, 0xC2, 0x08, 0x0 }; + +class ButtonWidget : Widget { + BitmapFont* font; + U8 text[1024]; + Context2D* image; + U32 color; +}; + +class MenuItemWidget : Widget { + BitmapFont* font; + U8 text[128]; + Context2D* icon; + U32 color; + U8* path; + Window* submenu; +}; + +class Context2DWidget : Widget { + Bool fast_copy; + Context2D* ctx; +}; + +class CheckBoxWidget : Widget { + Bool checked; +}; + +class RadioButtonWidget : Widget { + I64 group; + Bool selected; +}; + +class @terminal_widget_attr +{ + Bool bold; + Bool underline; + Bool blink; + Bool negative; + Bool invisible; +} + +class @terminal_widget_col +{ + U32 background; + U32 foreground; + U8 char; +}; + +class @terminal_widget_color +{ + U32 background; + U32 foreground; + U32 cursor; +}; + +class @terminal_widget_cursor +{ + I64 x; + I64 y; + Bool hidden; +}; + +class @terminal_widget_row +{ + @terminal_widget_col col[TERMINAL_MAX_COLS]; +}; + +class @terminal_widget_scroll +{ + I64 x; + I64 y; +}; + +class @terminal_widget_size +{ + I64 rows; + I64 cols; +}; + +class @terminal_widget_stored +{ + @terminal_widget_attr attr; + @terminal_widget_color color; + @terminal_widget_cursor cursor; +}; + +class TerminalWidget : Widget { + CFifoU8* input; + CFifoU8* output; + Bool refresh; + U8 consumed_chars[256]; + I64 state; + I64 last_fg_color_set; + @terminal_widget_attr attr; + @terminal_widget_color color; + @terminal_widget_cursor cursor; + @terminal_widget_row* row; + @terminal_widget_scroll scroll; + @terminal_widget_scroll max; + @terminal_widget_size size; + @terminal_widget_stored stored; +}; + +class BitmapFontTextInputWidget : Widget { + BitmapFont* font; + Bool blink; + Bool in_drag; + Bool is_password; + I64 cursor_index; + I64 mouse_drag_origin_x; + I64 mouse_drag_index; + I64 selected_region_start; + I64 selected_region_end; + I64 x_offset; + U32 color; + U8 text[1024]; + U8 password[1024]; + U0 (*SetFont)(U8* font); + U0 (*SetText)(U8* text); +} + +class BitmapFontTextLabelWidget : Widget { + BitmapFont* font; + U32 color; + U8 text[1024]; + U0 (*SetFont)(U8* font); + U0 (*SetText)(U8* text); +}; + +class TrueTypeTextInputWidget : Widget { + U8* font; + U32 color; + U32 bgcolor; + I64 size; + U8* text; + U8* prev_text; +}; + +class TrueTypeTextLabelWidget : Widget { + U8* font; + U32 color; + U32 bgcolor; + I64 size; + U8* text; + U8* prev_text; +}; + +class HorizontalSliderWidget : Widget { + Bool in_drag; + I64 scroll; + I64 max; + I64 value; +}; + +class VerticalSliderWidget : Widget { + Bool in_drag; + I64 scroll; + I64 max; + I64 value; +}; + +class HorizontalScrollBarWidget : Widget { + Bool in_drag; + I64 scroll; + I64 max; + I64 value; +}; + +class VerticalScrollBarWidget : Widget { + Bool in_drag; + I64 scroll; + I64 length; + I64 max; + I64 value; +}; + +class @list_view_item +{ + @list_view_item* prev; + @list_view_item* next; + Context2D* icon; + U8 text[1024]; +}; + +class ListViewWidget : Widget { + BitmapFont* font; + U32 color; + @list_view_item* items; +}; + +U0 @gui_widget_set_echo(Widget* widget, U8* echo) +{ + if (!widget || !echo) + return; + widget->echo = echo; +} + +U0 @gui_widget_set_font(Widget* widget, U8* font_name) +{ + if (!widget || !font_name) + return; + if (!StrLen(font_name)) + return; + switch (widget->type) { + case WIDGET_TYPE_BUTTON: + widget(ButtonWidget*)->font = BitmapFonts.GetByName(font_name); + break; + case WIDGET_TYPE_INPUT: + widget(BitmapFontTextInputWidget*)->font = BitmapFonts.GetByName(font_name); + break; + case WIDGET_TYPE_LABEL: + widget(BitmapFontTextLabelWidget*)->font = BitmapFonts.GetByName(font_name); + break; + case WIDGET_TYPE_LISTVIEW: + widget(ListViewWidget*)->font = BitmapFonts.GetByName(font_name); + break; + default: + break; + } +} + +U0 @gui_widget_set_mouse_pointer(Widget* widget, Context2D* pointer) +{ + if (!widget) + return; + widget->pointer = pointer; +} + +U0 @gui_widget_clear_mouse_pointer(Widget* widget) +{ + if (!widget) + return; + widget->pointer = NULL; +} + +U0 @gui_widget_set_opacity(Widget* widget, I64 opacity) +{ + if (!widget) + return; + widget->opacity = ClampI64(opacity, 0, 255); +} + +U0 @gui_widget_set_callback(Widget* widget, U8* name, U64 callback) +{ + if (!widget || !name || !callback) + return; + if (!StrCmp(name, "change")) + widget->callback.change = callback; + if (!StrCmp(name, "clicked")) + widget->callback.clicked = callback; + if (!StrCmp(name, "repaint")) + widget->callback.repaint = callback; +} + +U0 @gui_widget_set_text(Widget* widget, U8* text) +{ + if (!widget) + return; + switch (widget->type) { + case WIDGET_TYPE_BUTTON: + StrCpy(&widget(ButtonWidget*)->text, text); + break; + case WIDGET_TYPE_INPUT: + StrCpy(&widget(TextInputWidget*)->text, text); + break; + case WIDGET_TYPE_LABEL: + StrCpy(&widget(TextLabelWidget*)->text, text); + break; + case WIDGET_TYPE_MENU_ITEM: + StrCpy(&widget(MenuItemWidget*)->text, text); + break; + default: + break; + } +} + +Bool @widget_is_hovered(I64 x, I64 y, Widget* widget) +{ + if (Mouse.x > x && Mouse.x < x + widget->width && Mouse.y > y && Mouse.y < y + widget->height) + return TRUE; + return FALSE; +} + +Bool @gui_widget_is_hovered(Window* win, Widget* widget) +{ + return @widget_is_hovered(win->x + widget->x, win->y + widget->y, widget); +} + +U0 @widget_add_widget_to_list(Window* win, Widget* widget) +{ + if (!win || !widget) + return; + @window_widgets_list* widgets_list = win->widget; + while (widgets_list->next) { + widgets_list = widgets_list->next; + } + @window_widgets_list* widget_list_item = CAlloc(sizeof(@window_widgets_list)); + widget_list_item->widget = widget; + widget_list_item->prev = widgets_list; + widgets_list->next = widget_list_item; +} + +U0 @widget_input_backspace(BitmapFontTextInputWidget* widget) +{ + I64 i; + I64 len = StrLen(&widget->text); + for (i = widget->cursor_index - 1; i < len; i++) { + widget->text[i] = widget->text[i + 1]; + } +} + +U0 @widget_input_clear_selected_region(BitmapFontTextInputWidget* widget) +{ + widget->selected_region_start = -1; + widget->selected_region_end = -1; +} +Bool @widget_input_delete_selected_region(BitmapFontTextInputWidget* widget) +{ + I64 i; + I64 j; + if (widget->selected_region_start != -1 && widget->selected_region_end != -1) { + j = widget->selected_region_start; + for (i = widget->selected_region_end + 1; i < StrLen(&widget->text) + 1; + i++) { + widget->text[j++] = widget->text[i]; + } + widget->text[j] = NULL; + widget->cursor_index = widget->selected_region_start; + @widget_input_clear_selected_region(widget); + return TRUE; + } + return FALSE; +} + +U0 @widget_input_delete_at_cursor(BitmapFontTextInputWidget* widget) +{ + I64 i; + I64 len = StrLen(&widget->text); + for (i = widget->cursor_index; i < len; i++) + widget->text[i] = widget->text[i + 1]; + @widget_input_clear_selected_region(widget); +} + +U0 @widget_input_insert_char(BitmapFontTextInputWidget* widget, I64 char) +{ + U8 buf[1024]; + U8* pos; + I64 i, j, k; + j = 0; + for (i = widget->cursor_index; i < StrLen(&widget->text); i++) { + buf[j++] = widget->text[i]; + } + buf[j] = NULL; + for (i = widget->cursor_index; i < 1024; i++) { + widget->text[i] = NULL; + } + widget->text[StrLen(&widget->text)] = char; + pos = &buf; + while (*pos) { + widget->text[StrLen(&widget->text)] = *pos; + pos++; + } + @widget_input_clear_selected_region(widget); + widget->cursor_index++; +} + +U0 @widget_input_insert_scancode(BitmapFontTextInputWidget* widget, I64 key) +{ + U8 buf[1024]; + U8* pos; + I64 i, j, k; + j = 0; + for (i = widget->cursor_index; i < StrLen(&widget->text); i++) { + buf[j++] = widget->text[i]; + } + buf[j] = NULL; + for (i = widget->cursor_index; i < 1024; i++) { + widget->text[i] = NULL; + } + if (!Bt(kbd.down_bitmap, SC_SHIFT)) + widget->text[StrLen(&widget->text)] = NORMAL_KEY_SCAN_DECODE_TABLE(U8*)[key]; + else + widget->text[StrLen(&widget->text)] = SHIFT_KEY_SCAN_DECODE_TABLE(U8*)[key]; + pos = &buf; + while (*pos) { + widget->text[StrLen(&widget->text)] = *pos; + pos++; + } + @widget_input_clear_selected_region(widget); + widget->cursor_index++; +} + +U0 @widget_input_insert_text(BitmapFontTextInputWidget* widget, U8* text) +{ + while (*text) { + @widget_input_insert_char(widget, *text); + text++; + } +} + +Bool @widget_input_handle_key(BitmapFontTextInputWidget* widget) +{ + I64 key = Keyboard.active_key; + I64 tS = Keyboard.active_key_tS; + if (widget->cursor_index > StrLen(&widget->text) || widget->cursor_index < 0) + widget->cursor_index = 0; + if (widget->selected_region_start > widget->selected_region_end) { + @widget_input_clear_selected_region(widget); + } + if (key && tS != Keyboard.last_key_tS) { + switch (key) { + case SC_DELETE: + if (widget->selected_region_start != -1 && widget->selected_region_end != -1) { + @widget_input_delete_selected_region(widget); + } else { + @widget_input_delete_at_cursor(widget); + } + break; + case SC_HOME: + if (Bt(kbd.down_bitmap, SC_SHIFT) && widget->selected_region_start) { + widget->selected_region_start = 0; + if (widget->selected_region_end == -1) + widget->selected_region_end = widget->cursor_index - 1; + } else { + @widget_input_clear_selected_region(widget); + widget->cursor_index = 0; + } + break; + case SC_CURSOR_LEFT: + /* + "widget->selected_region_start : %d \n", widget->selected_region_start; + "widget->selected_region_end : %d \n", widget->selected_region_end; + "widget->cursor_index : %d \n", widget->cursor_index; + */ + if (widget->cursor_index) + widget->cursor_index--; + if (!Bt(kbd.down_bitmap, SC_SHIFT)) { + if (widget->selected_region_start != -1) + widget->cursor_index = widget->selected_region_start; + @widget_input_clear_selected_region(widget); + } else { + if (Bt(kbd.down_bitmap, SC_CTRL) && widget->selected_region_start) { + widget->selected_region_start = 0; + if (widget->selected_region_end == -1) + widget->selected_region_end = widget->cursor_index; + break; + } + switch (widget->selected_region_start) { + case -1: + widget->selected_region_start = widget->cursor_index; + widget->selected_region_end = widget->cursor_index; + break; + // case 0: + // break; + default: + if (widget->cursor_index > widget->selected_region_start) { + widget->selected_region_end = widget->cursor_index - 1; + } else { + widget->selected_region_start = widget->cursor_index; + if (widget->selected_region_start == widget->selected_region_end) { + @widget_input_clear_selected_region(widget); + } + } + break; + } + } + break; + case SC_END: + if (Bt(kbd.down_bitmap, SC_SHIFT)) { + widget->selected_region_start = widget->cursor_index; + widget->selected_region_end = StrLen(&widget->text) - 1; + } else { + @widget_input_clear_selected_region(widget); + widget->cursor_index = StrLen(&widget->text); + } + break; + case SC_CURSOR_RIGHT: + /* + "widget->selected_region_start : %d \n", widget->selected_region_start; + "widget->selected_region_end : %d \n", widget->selected_region_end; + "widget->cursor_index : %d \n", widget->cursor_index; + */ + if (!Bt(kbd.down_bitmap, SC_SHIFT)) { + if (widget->selected_region_end != -1) + widget->cursor_index = widget->selected_region_end; + @widget_input_clear_selected_region(widget); + } else { + if (Bt(kbd.down_bitmap, SC_CTRL)) { + widget->selected_region_start = widget->cursor_index; + widget->selected_region_end = StrLen(&widget->text) - 1; + break; + } + switch (widget->selected_region_start) { + case -1: + widget->selected_region_start = widget->cursor_index; + widget->selected_region_end = widget->cursor_index; + break; + default: + if (widget->cursor_index == widget->selected_region_start) { + widget->selected_region_start = widget->cursor_index + 1; + } else + widget->selected_region_end = widget->cursor_index; + break; + } + } + if (widget->cursor_index < StrLen(&widget->text)) + widget->cursor_index++; + break; + + case SC_BACKSPACE: + if (@widget_input_delete_selected_region(widget)) + return TRUE; + if (widget->cursor_index < 1) + break; + @widget_input_backspace(widget); + @widget_input_clear_selected_region(widget); + widget->cursor_index--; + break; + + case 0x02 ... 0x0D: + case 0x10 ... 0x1B: + case 0x1E ... 0x29: + case 0x2B ... 0x35: + case 0x39: + if (Bt(kbd.down_bitmap, SC_CTRL)) { + switch (ScanCode2Char(key)) { + case 'a': + if (StrLen(&widget->text)) { + widget->selected_region_start = 0; + widget->selected_region_end = StrLen(&widget->text) - 1; + } + break; + case 'c': + case 'x': + if (widget->selected_region_start != -1 && widget->selected_region_end != -1) { + U64 pos = &widget->text; + pos += widget->selected_region_start; + U8* text = StrNew(pos); + text[widget->selected_region_end - widget->selected_region_start + 1] = NULL; + Clipboard.Insert(CLIP_TYPE_TEXT, text); + if (ScanCode2Char(key) == 'x') + @widget_input_delete_selected_region(widget); + } + break; + case 'v': + // FIXME: Clipboard.Paste? + if (Clipboard.length) { + @widget_input_delete_selected_region(widget); + if (Clipboard.items->prev) { + if (Clipboard.items->prev->item->type == CLIP_TYPE_TEXT) { + @widget_input_insert_text( + widget, + Clipboard.items->prev->item(ClipboardTextItem*)->text); + } + } + } + break; + } + break; + } + @widget_input_delete_selected_region(widget); + @widget_input_insert_scancode(widget, key); + break; + default: + //@widget_input_delete_selected_region(widget); + break; + }; + return TRUE; + } + return FALSE; +} + +Widget* @widget_create_widget(Window* win, I64 type, I64 x, I64 y, I64 width, + I64 height) +{ + if (!win || !type) + return NULL; + + I64 size_of_widget; + Widget* widget; + + switch (type) { + case WIDGET_TYPE_NULL: + return NULL; + case WIDGET_TYPE_BUTTON: + size_of_widget = sizeof(ButtonWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_CHECKBOX: + size_of_widget = sizeof(CheckBoxWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_TERMINAL: + size_of_widget = sizeof(TerminalWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_RADIO: + size_of_widget = sizeof(RadioButtonWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_INPUT: + size_of_widget = sizeof(TextInputWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_LABEL: + size_of_widget = sizeof(TextLabelWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_LISTVIEW: + size_of_widget = sizeof(ListViewWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_MENU_ITEM: + size_of_widget = sizeof(MenuItemWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_TTF_LABEL: + size_of_widget = sizeof(TrueTypeTextLabelWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_CONTEXT2D: + size_of_widget = sizeof(Context2DWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_HORZ_SLIDER: + size_of_widget = sizeof(HorizontalSliderWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_VERT_SLIDER: + size_of_widget = sizeof(VerticalSliderWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_HORZ_SCROLLBAR: + size_of_widget = sizeof(HorizontalScrollBarWidget) * 2; + goto @widget_create_set_values; + case WIDGET_TYPE_VERT_SCROLLBAR: + size_of_widget = sizeof(VerticalScrollBarWidget) * 2; + goto @widget_create_set_values; + default: + return NULL; + } + @widget_create_set_values : widget = CAlloc(size_of_widget); + widget->type = type; + widget->x = x; + widget->y = y; + widget->width = width; + widget->height = height; + widget->parent_win = win; + + switch (type) { + case WIDGET_TYPE_TERMINAL: + widget(TerminalWidget*)->backing_store = NewContext2D(Display.Width(), Display.Height()); + widget(TerminalWidget*)->input = FifoU8New(65536); + widget(TerminalWidget*)->color.background = Color(0, 0, 0); + widget(TerminalWidget*)->color.foreground = Color(217, 217, 217); + widget(TerminalWidget*)->color.cursor = Color(217, 217, 0); + widget(TerminalWidget*)->row = CAlloc(sizeof(@terminal_widget_row) * 2000); + break; + case WIDGET_TYPE_INPUT: + widget(TextInputWidget*)->color = Color(0, 0, 0); + widget(TextInputWidget*)->cursor_index = -1; + widget(TextInputWidget*)->mouse_drag_index = -1; + widget(TextInputWidget*)->selected_region_start = -1; + widget(TextInputWidget*)->selected_region_end = -1; + break; + case WIDGET_TYPE_LABEL: + widget(TextLabelWidget*)->color = Color(0, 0, 0); + widget(TextLabelWidget*)->SetText = CAlloc(0x1F, Fs->code_heap); + I32 addr = widget(TextLabelWidget*)->SetText; + MemCpy(addr, widget_self_set1, 0x1F); + MemCpy(addr + 12, &widget, 8); + Function.InsertCall(addr + 21, Gui.Widget.SetText); + break; + case WIDGET_TYPE_LISTVIEW: + widget(ListViewWidget*)->color = Color(0, 0, 0); + widget(ListViewWidget*)->items = CAlloc(sizeof(@list_view_item)); + break; + } + + @widget_add_widget_to_list(win, widget); + return widget; +} + +U0 @widget_init_widget(Widget* widget, Window* win, I64 type, I64 x, I64 y, + I64 width, I64 height) +{ + if (!win || !widget || !type) + return; + + I64 size_of_widget = 0; + I32 addr = NULL; + + switch (type) { + case WIDGET_TYPE_NULL: + return; + case WIDGET_TYPE_BUTTON: + size_of_widget = sizeof(ButtonWidget); + break; + case WIDGET_TYPE_CHECKBOX: + size_of_widget = sizeof(CheckBoxWidget); + break; + case WIDGET_TYPE_TERMINAL: + size_of_widget = sizeof(TerminalWidget); + break; + case WIDGET_TYPE_RADIO: + size_of_widget = sizeof(RadioButtonWidget); + break; + case WIDGET_TYPE_INPUT: + size_of_widget = sizeof(TextInputWidget); + break; + case WIDGET_TYPE_LABEL: + size_of_widget = sizeof(TextLabelWidget); + break; + case WIDGET_TYPE_LISTVIEW: + size_of_widget = sizeof(ListViewWidget); + break; + case WIDGET_TYPE_MENU_ITEM: + size_of_widget = sizeof(MenuItemWidget); + break; + case WIDGET_TYPE_TTF_LABEL: + size_of_widget = sizeof(TrueTypeTextLabelWidget); + break; + case WIDGET_TYPE_CONTEXT2D: + size_of_widget = sizeof(Context2DWidget); + break; + case WIDGET_TYPE_HORZ_SLIDER: + size_of_widget = sizeof(HorizontalSliderWidget); + break; + case WIDGET_TYPE_VERT_SLIDER: + size_of_widget = sizeof(VerticalSliderWidget); + break; + case WIDGET_TYPE_HORZ_SCROLLBAR: + size_of_widget = sizeof(HorizontalScrollBarWidget); + break; + case WIDGET_TYPE_VERT_SCROLLBAR: + size_of_widget = sizeof(VerticalScrollBarWidget); + break; + default: + break; + } + + MemSet(widget, NULL, size_of_widget); + + widget->type = type; + widget->x = x; + widget->y = y; + widget->width = width; + widget->height = height; + widget->parent_win = win; + + switch (type) { + case WIDGET_TYPE_TERMINAL: + widget(TerminalWidget*)->backing_store = NewContext2D(Display.Width(), Display.Height()); + widget(TerminalWidget*)->input = FifoU8New(65536); + widget(TerminalWidget*)->color.background = Color(0, 0, 0); + widget(TerminalWidget*)->color.foreground = Color(217, 217, 217); + widget(TerminalWidget*)->color.cursor = Color(217, 217, 0); + widget(TerminalWidget*)->row = CAlloc(sizeof(@terminal_widget_row) * 2000); + break; + case WIDGET_TYPE_INPUT: + widget(TextInputWidget*)->color = Color(0, 0, 0); + widget(TextInputWidget*)->cursor_index = -1; + widget(TextInputWidget*)->mouse_drag_index = -1; + widget(TextInputWidget*)->selected_region_start = -1; + widget(TextInputWidget*)->selected_region_end = -1; + widget(TextInputWidget*)->color = Color(0, 0, 0); + widget(TextInputWidget*)->SetText = CAlloc(0x1F, Fs->code_heap); + addr = widget(TextInputWidget*)->SetFont; + MemCpy(addr, widget_self_set1, 0x1F); + MemCpy(addr + 12, &widget, 8); + Function.InsertCall(addr + 21, Gui.Widget.SetFont); + addr = widget(TextInputWidget*)->SetText; + MemCpy(addr, widget_self_set1, 0x1F); + MemCpy(addr + 12, &widget, 8); + Function.InsertCall(addr + 21, Gui.Widget.SetText); + break; + case WIDGET_TYPE_LABEL: + widget(TextLabelWidget*)->color = Color(0, 0, 0); + widget(TextLabelWidget*)->SetText = CAlloc(0x1F, Fs->code_heap); + addr = widget(TextLabelWidget*)->SetFont; + MemCpy(addr, widget_self_set1, 0x1F); + MemCpy(addr + 12, &widget, 8); + Function.InsertCall(addr + 21, Gui.Widget.SetFont); + addr = widget(TextLabelWidget*)->SetText; + MemCpy(addr, widget_self_set1, 0x1F); + MemCpy(addr + 12, &widget, 8); + Function.InsertCall(addr + 21, Gui.Widget.SetText); + break; + case WIDGET_TYPE_LISTVIEW: + widget(ListViewWidget*)->color = Color(0, 0, 0); + widget(ListViewWidget*)->items = CAlloc(sizeof(@list_view_item)); + break; + } + + @widget_add_widget_to_list(win, widget); +} + +Gui.CreateWidget = &@widget_create_widget; +Gui.InitWidget = &@widget_init_widget; + +Gui.Widget.IsHovered = &@gui_widget_is_hovered; +Gui.Widget.SetCallback = &@gui_widget_set_callback; +Gui.Widget.SetEcho = &@gui_widget_set_echo; +Gui.Widget.SetFont = &@gui_widget_set_font; +Gui.Widget.SetMousePointer = &@gui_widget_set_mouse_pointer; +Gui.Widget.ClearMousePointer = &@gui_widget_clear_mouse_pointer; +Gui.Widget.SetOpacity = &@gui_widget_set_opacity; +Gui.Widget.SetText = &@gui_widget_set_text; + +"widget "; \ No newline at end of file diff --git a/System/MakeSystem.HC b/System/MakeSystem.HC new file mode 100644 index 0000000..6d9d815 --- /dev/null +++ b/System/MakeSystem.HC @@ -0,0 +1,103 @@ +/* clang-format off */ + +DocMax(adam_task); +WinMax(adam_task); +WinToTop(adam_task); + +#include "Setup/Environment"; + +// Erythros system drivers +"drivers: { "; +#include "Drivers/Audio"; +#include "Drivers/Display"; +#include "Drivers/Mouse"; +#include "Drivers/Pci"; +#include "Drivers/Virtio-blk"; +#include "Drivers/VMSVGA"; +#include "Drivers/VMwareTools"; +#include "Drivers/AC97"; +"}\n"; + +// FFI support files +#include "FFI/Base"; +#include "FFI/LibC"; +#include "FFI/New"; +#include "FFI/ELF64"; + +// stb_image library +#include "Utilities/Image"; +load_elf("M:/build/bin/image"); + +// Jakt support files +#include "Jakt/OS"; +#include "Jakt/IOPort"; +#include "Jakt/PCI"; +#include "Jakt/Time"; + +#include "Libraries/Tlse"; +load_elf("M:/build/bin/tlse"); + +// Networking APIs +#include "Api/Dns.HC"; +#include "Api/Icmp.HC"; +#include "Api/Ipv4.HC"; +#include "Api/MD5.HC"; +#include "Api/NetInfo.HC"; +#include "Api/Tcp.HC"; +#include "Api/Tls.HC"; + +// Erythros system libraries +"libraries: { "; +#include "Libraries/Function"; +#include "Libraries/Base64"; +#include "Libraries/String"; +#include "Libraries/BitmapFont"; +#include "Libraries/Display"; +#include "Libraries/FileSystem"; +#include "Libraries/Graphics2D"; +#include "Libraries/Animation2D"; +#include "Libraries/Image"; +#include "Libraries/Json"; +#include "Libraries/Rsa"; +#include "Libraries/Sha256"; +#include "Libraries/System"; +#include "Libraries/RawText"; +#include "Libraries/User"; +#include "Libraries/Session"; +#include "Libraries/Shell"; +#include "Libraries/Stdio"; +#include "Libraries/Http"; +#include "Libraries/Audio"; +#include "Libraries/Gui"; +#include "Libraries/Ipc"; +#include "Libraries/Clipboard"; +#include "Libraries/Widget"; +#include "Libraries/Theme"; +"}\n"; + +@http_init_tmp_and_cache_directories; + +load_elf("M:/build/bin/net"); + +// Networking Utilities +#include "Utilities/Dns"; +#include "Utilities/NetRep"; +#include "Utilities/Ping"; +#include "Utilities/Time"; + +Spawn(_start, , "Net Task"); + +TimeSync; + +#include "Core/Compositor"; +#include "Core/FileSystem"; +#include "Core/Menu"; +#include "Core/MessageBox"; +#include "Core/Shell"; +#include "Core/ShellCommands"; +#include "Core/SystemTray"; +#include "Core/SystemStarter"; + +#include "Setup/Init"; + +/* clang-format on */ diff --git a/System/Setup/Environment.HC b/System/Setup/Environment.HC new file mode 100644 index 0000000..506a661 --- /dev/null +++ b/System/Setup/Environment.HC @@ -0,0 +1,229 @@ +AutoComplete(0); + +#define include_noreindex #include + +I64 tos_nist_offset = 5603; // UTC -4 + +#define NIST_TIME_OFFSET (tos_nist_offset - local_time_offset / CDATE_FREQ) + +public +I64 CDate2Unix(CDate dt) +{ // TempleOS datetime to Unix timestamp. + return ToI64((dt - Str2Date("1/1/1970")) / CDATE_FREQ + NIST_TIME_OFFSET); +} + +public +CDate Unix2CDate(I64 timestamp) +{ // Unix timestamp to TempleOS datetime. + return (timestamp - NIST_TIME_OFFSET) * CDATE_FREQ + Str2Date("1/1/1970"); +} + +// FIXME: Put these in a "Builtin" library? +U0 FifoU8Cpy(CFifoU8* f, U8* s) +{ + if (!f || !s) + return; + while (*s) + FifoU8Ins(f, *s++); +} +Bool KeyDown(I64 sc) return Bt(kbd.down_bitmap, sc); +I64 T(Bool _condition, I64 _true, I64 _false) +{ + if (_condition) + return _true; + return _false; +} + +asm + { +_MEMCPY_U16:: + PUSH RBP + MOV RBP,RSP + PUSH RSI + PUSH RDI + CLD + MOV RDI,U64 SF_ARG1[RBP] + MOV RSI,U64 SF_ARG2[RBP] + MOV RCX,U64 SF_ARG3[RBP] + REP_MOVSW + MOV RAX,RDI + POP RDI + POP RSI + POP RBP + RET1 24 +_MEMCPY_U32:: + PUSH RBP + MOV RBP,RSP + PUSH RSI + PUSH RDI + CLD + MOV RDI,U64 SF_ARG1[RBP] + MOV RSI,U64 SF_ARG2[RBP] + MOV RCX,U64 SF_ARG3[RBP] + REP_MOVSD + MOV RAX,RDI + POP RDI + POP RSI + POP RBP + RET1 24 +_MEMCPY_U64:: + PUSH RBP + MOV RBP,RSP + PUSH RSI + PUSH RDI + CLD + MOV RDI,U64 SF_ARG1[RBP] + MOV RSI,U64 SF_ARG2[RBP] + MOV RCX,U64 SF_ARG3[RBP] + REP_MOVSQ + MOV RAX,RDI + POP RDI + POP RSI + POP RBP + RET1 24 + } + +public _extern _MEMCPY_U16 U16* MemCpyU16(U16* dst, U16* src, I64 cnt); +public +_extern _MEMCPY_U32 U32* MemCpyU32(U32* dst, U32* src, I64 cnt); +public +_extern _MEMCPY_U64 U64* MemCpyU64(U64* dst, U64* src, I64 cnt); + +I64 @lerp(U32 val, U32 mx1, U32 mx2) +{ + F64 r = (val & mx1) / ToF64(mx1); + return ToI64(r * mx2); +} + +U0 @patch_call_rel32(U32 from, U32 to) +{ + *(from(U8*)) = 0xE8; + *((from + 1)(I32*)) = to - from - 5; +} + +U0 @patch_jmp_rel32(U32 from, U32 to) +{ + *(from(U8*)) = 0xE9; + *((from + 1)(I32*)) = to - from - 5; +} + +CMemBlk* ShrinkMemBlkByPags(CMemBlk* from, I64 count) +{ + from->pags -= count; + U64 to = from; + to += count * MEM_PAG_SIZE; + MemCpy(to, from, MEM_PAG_SIZE); + return to; +} + +U0 @sse_enable() +{ + /* clang-format off */ + asm + { + MOV_EAX_CR0 + AND AX, 0xFFFB // clear coprocessor emulation CR0.EM + OR AX, 0x2 // set coprocessor monitoring CR0.MP + MOV_CR0_EAX + MOV_EAX_CR4 + OR AX, 3 << 9 // set CR4.OSFXSR and CR4.OSXMMEXCPT at the same time + MOV_CR4_EAX + } + /* clang-format on */ +} + +U0 @sse_enable_on_all_cores() +{ + I64 i; + for (i = 1; i < mp_cnt; i++) + Spawn(&@sse_enable, , , i); +} + +I64 @t(Bool _condition, I64 _true, I64 _false) +{ + if (_condition) + return _true; + return _false; +} + +U0 @erythros_mem_task_loop() +{ + while (1) { + Sleep(1); + }; +} + +// Before doing anything else, we: + +// 1. Mark memory in code heap below 0x1000000 as used. +sys_code_bp->mem_free_lst->next->pags = 0; + +// 2. Free up 64MB at bottom of code heap for non-HolyC programs +sys_code_bp->mem_free_lst = ShrinkMemBlkByPags(sys_code_bp->mem_free_lst, 131072); + +// 3. Enable SSE +@sse_enable; + +// 4. Init mem_tasks + +CTask* erythros_mem_task = Spawn(&@erythros_mem_task_loop, , "ErythrosMemTask"); + +#define MALLOC_MEM_TASK_COUNT 4 +CTask** malloc_mem_task = CAlloc(sizeof(CTask*) * MALLOC_MEM_TASK_COUNT, erythros_mem_task); +I64 malloc_current_mem_task = 0; + +U0 @malloc_mem_tasks_init() +{ + U8* scratch_buffer[64]; + I64 i; + for (i = 0; i < MALLOC_MEM_TASK_COUNT; i++) { + StrPrint(scratch_buffer, "ErythrosMallocTask%d", i); + malloc_mem_task[i] = Spawn(&@erythros_mem_task_loop, , scratch_buffer); + } +} + +@malloc_mem_tasks_init; + +U0 dd() { DocDump(adam_task->put_doc); } +//@patch_jmp_rel32(&Fault2, &Reboot); // Reboot instead of crashing to the debugger +U0 NoBeep(I8, Bool) {}; +@patch_jmp_rel32(&Beep, &NoBeep); // Don't delay on beep when entering debugger + +Bool BlkDevLock2(CBlkDev* bd) +{ + BlkDevChk(bd); + while (bd->lock_fwding) + bd = bd->lock_fwding; + if (!Bt(&bd->locked_flags, BDlf_LOCKED) || bd->owning_task != Fs) { + while (LBts(&bd->locked_flags, BDlf_LOCKED)) + Sleep(Rand * 10); + bd->owning_task = Fs; + return TRUE; + } else + return FALSE; +} + +Bool DrvLock2(CDrv* dv) +{ + DrvChk(dv); + BlkDevLock2(dv->bd); + if (!Bt(&dv->locked_flags, DVlf_LOCKED) || dv->owning_task != Fs) { + while (LBts(&dv->locked_flags, DVlf_LOCKED)) + Sleep(Rand * 10); + dv->owning_task = Fs; + return TRUE; + } else + return FALSE; +} + +@patch_jmp_rel32(&BlkDevLock, &BlkDevLock2); // Patch BlkDevLock so we don't deadlock on multiple tasks reading from virtio disk +@patch_jmp_rel32(&DrvLock, &DrvLock2); // Patch DrvLock so we don't deadlock on multiple tasks reading from virtio disk + +U0 @erythros_mem_task_loop() +{ + while (1) { + Sleep(1); + }; +} + +CTask* erythros_mem_task = Spawn(&@erythros_mem_task_loop, , "ErythrosMemTask"); diff --git a/System/Setup/Init.HC b/System/Setup/Init.HC new file mode 100644 index 0000000..8738cef --- /dev/null +++ b/System/Setup/Init.HC @@ -0,0 +1,65 @@ + +// Save pointer to TempleOS system-wide (CTRL-ALT) callbacks +U64 tos_fp_cbs_enabled = keydev.fp_ctrl_alt_cbs; +U64 tos_fp_cbs_disabled = CAlloc(0xD0); + +U0 @erythros_init() +{ + I64 err = 0; + + // Initialize Display + if (!Display.Driver()) { + err = Display.Init(1920, 1080, 32, FB_VMSVGA); + } + + if (err) { + DocClear(Fs->put_doc); + "No supported display device found."; + while (1) { + Sleep(1); + }; + } + + // Initialize Mouse + Mouse.Init(); + Spawn(Mouse.Task, , "Mouse"); + + // Enable debug output + Raw(ON); + DocDump(adam_task->put_doc); + + // Disable TempleOS system-wide (CTRL-ALT) callbacks + keydev.fp_ctrl_alt_cbs = tos_fp_cbs_disabled; + + // Suspend TempleOS Window Manager task + Suspend(sys_winmgr_task); + + // Reassign VGA writes to a random buffer to avoid collision with SVGA FB + text.vga_alias = MAlloc(1048576, adam_task); + + // Initialize Graphics2D Library + Graphics2D.Init(); + + "\x1b[2J\x1b[H"; + //"%s\n", System.BuildInfo(); + + switch (Display.Driver()) { + case FB_VMSVGA: + "Display driver is: VMSVGA\n"; + break; + } + + // Initialize FileSystem + // FileSystem.Init(); + + // Initialize Compositor + Compositor.Init(); + + // Spawn Compositor + Spawn(Compositor.Task, , "Compositor"); + + // Spawn SystemStarter + Spawn(SystemStarter.Task, , "SystemStarter", 1); +} + +@erythros_init; diff --git a/System/Shell/Commands/aplay.HC b/System/Shell/Commands/aplay.HC new file mode 100644 index 0000000..274c4b2 --- /dev/null +++ b/System/Shell/Commands/aplay.HC @@ -0,0 +1,35 @@ +I64 @shell_cmd_aplay(@shell* sh, I64 argc, U8** argv) +{ + U8 buf[512]; + if (argc < 2) { + StrPrint(&buf, "Error reading file %s\n", argv[1]); + Stdio.WriteLine(sh, "Usage: aplay [OPTION]... [FILE]...\n"); + return 1; + } + I64 i; + I64 size; + Sound* snd; + U8* filename = NULL; + for (i = 1; i < argc; i++) { + filename = @shell_expand_relative_path(sh, argv[i]); + System.Log(Fs, "filename: %s", filename); + if (FileSystem.PathExists(filename)) { + snd = Audio.SoundFromFile(filename); + if (!snd) { + StrPrint(&buf, "%s: Error playing audio file\n", filename); + Stdio.WriteLine(sh, &buf); + Free(filename); + return 1; + } + Audio.PlaySound(snd); + Audio.FreeSound(snd); + Free(filename); + } else { + StrPrint(&buf, "%s: No such file or directory\n", filename); + Stdio.WriteLine(sh, &buf); + Free(filename); + return 1; + } + } + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/cat.HC b/System/Shell/Commands/cat.HC new file mode 100644 index 0000000..4d4c9fc --- /dev/null +++ b/System/Shell/Commands/cat.HC @@ -0,0 +1,19 @@ +I64 @shell_cmd_cat(@shell* sh, I64 argc, U8** argv) +{ + if (argc < 2) + return 0; + I64 i; + I64 j; + I64 size = 0; + U8* filename = NULL; + U8* buf = NULL; + for (i = 1; i < argc; i++) { + filename = @shell_expand_relative_path(sh, argv[i]); + buf = FileSystem.ReadFile(filename, &size); + for (j = 0; j < size; j++) + FifoU8Ins(sh->output, buf[j]); + Free(buf); + Free(filename); + } + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/clear.HC b/System/Shell/Commands/clear.HC new file mode 100644 index 0000000..11afcf9 --- /dev/null +++ b/System/Shell/Commands/clear.HC @@ -0,0 +1,9 @@ +I64 @shell_cmd_clear(@shell* sh, I64 argc, U8** argv) +{ + if (argc > 1) { + Stdio.WriteLine(sh, "esh: clear: too many arguments\n"); + return 1; + } + Stdio.WriteLine(sh, "\x1b[2J\x1b[0;0H"); + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/echo.HC b/System/Shell/Commands/echo.HC new file mode 100644 index 0000000..5cbe7d0 --- /dev/null +++ b/System/Shell/Commands/echo.HC @@ -0,0 +1,10 @@ +I64 @shell_cmd_echo(@shell* sh, I64 argc, U8** argv) +{ + I64 i; + for (i = 1; i < argc; i++) { + Stdio.WriteLine(sh, argv[i]); + Stdio.WriteLine(sh, " "); + } + Stdio.WriteLine(sh, "\n"); + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/esh.HC b/System/Shell/Commands/esh.HC new file mode 100644 index 0000000..8d129aa --- /dev/null +++ b/System/Shell/Commands/esh.HC @@ -0,0 +1,5 @@ +I64 @shell_cmd_esh(@shell* sh, I64 argc, U8** argv) +{ + @shell_input_loop(sh); + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/history.HC b/System/Shell/Commands/history.HC new file mode 100644 index 0000000..a9185a1 --- /dev/null +++ b/System/Shell/Commands/history.HC @@ -0,0 +1,14 @@ +I64 @shell_cmd_history(@shell* sh, I64 argc, U8** argv) +{ + I64 i; + I64 j; + U8 buf[512]; + for (i = 0; i < sh->history.pos; i++) { + StrPrint(&buf, "%05d %s\n", i + 1, sh->history.entries[i]); + j = 0; + while (buf[j] == '0') + buf[j++] = ' '; + Stdio.WriteLine(sh, &buf); + } + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/ifconfig.HC b/System/Shell/Commands/ifconfig.HC new file mode 100644 index 0000000..3da268e --- /dev/null +++ b/System/Shell/Commands/ifconfig.HC @@ -0,0 +1,47 @@ +#define ETHERNET_FRAME_SIZE 1400 + +I64 @shell_cmd_ifconfig(@shell* sh, I64 argc, U8** argv) +{ + NetInfoRequest* req = @net_info_request; + + U64 en0_mac = req->mac_address; + U32 en0_addr = req->ipv4_address; + U32 en0_mask = req->ipv4_netmask; + U32 en0_bcast = req->ipv4_address | ~req->ipv4_netmask; + + U8 buf[512]; + + Stdio.WriteLine(sh, "en0: flags=4163 mtu %d\n", + ETHERNET_FRAME_SIZE - 18); + + Stdio.WriteLine(sh, + " inet %d.%d.%d.%d netmask %d.%d.%d.%d broadcast " + "%d.%d.%d.%d\n", + en0_addr.u8[3], en0_addr.u8[2], en0_addr.u8[1], en0_addr.u8[0], + en0_mask.u8[3], en0_mask.u8[2], en0_mask.u8[1], en0_mask.u8[0], + en0_bcast.u8[3], en0_bcast.u8[2], en0_bcast.u8[1], en0_bcast.u8[0]); + + Stdio.WriteLine(sh, + " ether %02x:%02x:%02x:%02x:%02x:%02x txqueuelen 0 " + "(Ethernet)\n", + en0_mac.u8[5], en0_mac.u8[4], en0_mac.u8[3], en0_mac.u8[2], en0_mac.u8[1], + en0_mac.u8[0]); + + Stdio.WriteLine(sh, " RX packets %d bytes %d\n", req->rx_frames, + req->rx_bytes); + + Stdio.WriteLine(sh, " RX errors %d dropped %d overruns %d frame %d\n", 0, + 0, 0, 0); // TODO + + Stdio.WriteLine(sh, " TX packets %d bytes %d\n", req->tx_frames, + req->tx_bytes); + + Stdio.WriteLine(sh, + " TX errors %d dropped %d overruns %d carrier %d " + "collisions %d\n", + 0, 0, 0, 0, 0); // TODO + + Stdio.WriteLine(sh, "\n"); + + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/ls.HC b/System/Shell/Commands/ls.HC new file mode 100644 index 0000000..4b19a44 --- /dev/null +++ b/System/Shell/Commands/ls.HC @@ -0,0 +1,70 @@ +#define @shell_cmd_ls_opt_a 0 +#define @shell_cmd_ls_opt_l 1 + +I64 @shell_cmd_ls_output(@shell* sh, U8* arg_path, I64 flags) +{ + U8 buf[512]; + U8* path = @shell_expand_relative_path(sh, arg_path); + if (!FileSystem.PathExists(path)) { + StrPrint(&buf, "ls: cannot access '%s': No such file or directory\n", path); + Stdio.WriteLine(sh, &buf); + Free(path); + return 2; + } + @dir_entry* tmpf = NULL; + @dir_entry* tmpf2 = NULL; + tmpf = FileSystem.GetFiles(path); + if (tmpf) + while (tmpf) { + if (tmpf->type == DE_TYPE_DIR) { + StrCpy(&buf, "\x1b[1;34m"); + Stdio.WriteLine(sh, &buf); + } + StrPrint(&buf, "%s\x1b[0m %u\n", &tmpf->name, tmpf->size); + Stdio.WriteLine(sh, &buf); + tmpf2 = tmpf; + tmpf = tmpf->next; + Free(tmpf2); + } + Free(path); + return 0; +} + +I64 @shell_cmd_ls(@shell* sh, I64 argc, U8** argv) +{ + U8 buf[512]; + U8* options_list = "al"; + U64 options_err = NULL; + I64 dir_cnt = 0; + I64 flags = NULL; + I64 rval = 0; + I64 i; + switch (@shell_parse_opts(sh, options_list, argc, argv, &flags, &options_err, + TRUE)) { + case SHELL_OPTS_ERR_INVALID_OPT: + StrPrint(&buf, "ls: invalid option -- '%s'\n", options_err); + Stdio.WriteLine(sh, &buf); + break; + default: + break; + } + if (options_err) { + Free(options_err); + return 2; + } + + for (i = 1; i < argc; i++) + if (argv[i][0] != '-') + dir_cnt++; + + if (!dir_cnt) { + return @shell_cmd_ls_output(sh, &sh->cwd, flags); + } else { + for (i = 1; i < argc; i++) + if (argv[i][0] != '-') { + rval = Max(rval, @shell_cmd_ls_output(sh, argv[i], flags)); + } + } + + return rval; +} \ No newline at end of file diff --git a/System/Shell/Commands/nslookup.HC b/System/Shell/Commands/nslookup.HC new file mode 100644 index 0000000..ee71f71 --- /dev/null +++ b/System/Shell/Commands/nslookup.HC @@ -0,0 +1,32 @@ +I64 @shell_cmd_nslookup(@shell* sh, I64 argc, U8** argv) +{ + if (argc < 2) { + // TODO: Interactive mode + return 0; + } + if (argc > 2) { + // TODO: Server argument + } + + NetInfoRequest* req = @net_info_request; + U32 resolver_ip = req->dns_server_address; + Free(req); + + Stdio.WriteLine(sh, "Server: %d.%d.%d.%d\n", resolver_ip.u8[3], + resolver_ip.u8[2], resolver_ip.u8[1], resolver_ip.u8[0]); + + Stdio.WriteLine(sh, "Address: %d.%d.%d.%d#53\n\n", resolver_ip.u8[3], + resolver_ip.u8[2], resolver_ip.u8[1], resolver_ip.u8[0]); + + U32 res_ip = @dns_query(argv[1]); + + if (res_ip == U32_MAX) { + Stdio.WriteLine(sh, "** server can't find %s: NXDOMAIN\n\n", argv[1]); + return 1; + } + Stdio.WriteLine(sh, "Non-authoritative answer:\n"); + Stdio.WriteLine(sh, "Name: %s\n", argv[1]); + Stdio.WriteLine(sh, "Address: %d.%d.%d.%d\n\n", res_ip.u8[3], res_ip.u8[2], + res_ip.u8[1], res_ip.u8[0]); + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/open.HC b/System/Shell/Commands/open.HC new file mode 100644 index 0000000..e763db1 --- /dev/null +++ b/System/Shell/Commands/open.HC @@ -0,0 +1,8 @@ +I64 @shell_cmd_open(@shell* sh, I64 argc, U8** argv) +{ + if (argc < 2) { + Stdio.WriteLine(sh, "open: path required\n"); + return 1; + } + return @systemstarter_open(sh, argc, argv); +} \ No newline at end of file diff --git a/System/Shell/Commands/ping.HC b/System/Shell/Commands/ping.HC new file mode 100644 index 0000000..854a676 --- /dev/null +++ b/System/Shell/Commands/ping.HC @@ -0,0 +1,58 @@ +I64 @shell_cmd_ping(@shell* sh, I64 argc, U8** argv) +{ + if (argc < 2) { + Stdio.WriteLine(sh, "ping: usage error: Destination address required\n"); + return 1; + } + U8* host = argv[1]; + if (!host || !StrLen(host)) { + Stdio.WriteLine(sh, "Invalid host specified\n"); + return PING_ERR_INVALID_HOST; + } + I64 count = 4; + + U32 addr = @dns_query(host); + if (addr == U32_MAX) { + Stdio.WriteLine(sh, "Host not found\n"); + return PING_ERR_HOST_NOT_FOUND; + } + + U16 iden = (RandU16 * SysTimerRead) & 0xFFFF; + I64 start_jiffies; + U32 reply = NULL; + I64 res = 0; + U16 seq = 0; + I64 loss = 0; + + IcmpRequest* request = CAlloc(sizeof(IcmpRequest), Fs->code_heap); + + Stdio.WriteLine(sh, "PING %s (%d.%d.%d.%d): %d data bytes\n", + host, addr.u8[3], addr.u8[2], addr.u8[1], addr.u8[0], PING_PAYLOAD_SIZE); + + I64 i; + for (i = 0; i < count; i++) { + start_jiffies = cnts.jiffies; + reply = @icmp_echo_request(addr, iden, seq, request, i); + if (!reply) { + Stdio.WriteLine(sh, "Request timeout for icmp_seq %d\n", seq); + ++loss; + res = 1; + } else { + Stdio.WriteLine(sh, "%d bytes from %d.%d.%d.%d: icmp_seq=%d ttl=%d time=%d ms\n", + reply.u16[1], addr.u8[3], addr.u8[2], addr.u8[1], addr.u8[0], seq, reply.u16[0], cnts.jiffies - start_jiffies); + } + while (cnts.jiffies < start_jiffies + 1000 && i < (count - 1)) + Sleep(1); + ++seq; + } + + Free(request); + + Stdio.WriteLine(sh, "--- %d.%d.%d.%d ping statistics ---\n", addr.u8[3], addr.u8[2], addr.u8[1], addr.u8[0]); + Stdio.WriteLine(sh, "%d packets transmitted, %d packets received, %0f", + seq, seq - loss, (loss * 1.0 / seq * 1.0) * 100); + Stdio.WriteLine(sh, "%c", 37); + Stdio.WriteLine(sh, " packet loss\n"); + + return res; +} diff --git a/System/Shell/Commands/poweroff.HC b/System/Shell/Commands/poweroff.HC new file mode 100644 index 0000000..7e0073a --- /dev/null +++ b/System/Shell/Commands/poweroff.HC @@ -0,0 +1,5 @@ +I64 @shell_cmd_poweroff(@shell* sh, I64 argc, U8** argv) +{ + System.PowerOff(); + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/pwd.HC b/System/Shell/Commands/pwd.HC new file mode 100644 index 0000000..460cbad --- /dev/null +++ b/System/Shell/Commands/pwd.HC @@ -0,0 +1,7 @@ +I64 @shell_cmd_pwd(@shell* sh, I64 argc, U8** argv) +{ + U8 buf[512]; + StrPrint(&buf, "%s\n", &sh->cwd); + Stdio.WriteLine(sh, &buf); + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/reboot.HC b/System/Shell/Commands/reboot.HC new file mode 100644 index 0000000..2bec99a --- /dev/null +++ b/System/Shell/Commands/reboot.HC @@ -0,0 +1,5 @@ +I64 @shell_cmd_reboot(@shell* sh, I64 argc, U8** argv) +{ + Reboot; + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/sh.HC b/System/Shell/Commands/sh.HC new file mode 100644 index 0000000..b564a59 --- /dev/null +++ b/System/Shell/Commands/sh.HC @@ -0,0 +1,4 @@ +I64 @shell_cmd_sh(@shell* sh, I64 argc, U8** argv) +{ + return @shell_cmd_esh(sh, argc, argv); +} \ No newline at end of file diff --git a/System/Shell/Commands/uname.HC b/System/Shell/Commands/uname.HC new file mode 100644 index 0000000..900254d --- /dev/null +++ b/System/Shell/Commands/uname.HC @@ -0,0 +1,81 @@ +#define @shell_cmd_uname_opt_s 0 +#define @shell_cmd_uname_opt_n 1 +#define @shell_cmd_uname_opt_r 2 +#define @shell_cmd_uname_opt_v 3 +#define @shell_cmd_uname_opt_m 4 +#define @shell_cmd_uname_opt_p 5 +#define @shell_cmd_uname_opt_i 6 +#define @shell_cmd_uname_opt_o 7 +#define @shell_cmd_uname_opt_a 8 + +I64 @shell_cmd_uname(@shell* sh, I64 argc, U8** argv) +{ + I64 i; + CDateStruct* ds = CAlloc(sizeof(CDateStruct)); + Date2Struct(ds, sys_compile_time); + U8* options_list = "snrvmpioa"; + U64 options_err = NULL; + U8* ds_m = "JanFebMarAprMayJunJulAugSepOctNovDec"; + U8* ds_d = "SunMonTueWedThuFriSat"; + U8* ds_mm = " "; + U8* ds_dd = " "; + U8 buf[512]; + I64 flags = NULL; + StrCpy(&buf, ""); + if (argc < 2) + flags |= 1 << @shell_cmd_uname_opt_s; + switch ( + @shell_parse_opts(sh, options_list, argc, argv, &flags, &options_err)) { + case SHELL_OPTS_ERR_INVALID_OPT: + StrPrint(&buf, "uname: invalid option -- '%s'\n", options_err); + Stdio.WriteLine(sh, &buf); + break; + case SHELL_OPTS_ERR_EXTRA_OPD: + StrPrint(&buf, "uname: extra operand '%s'\n", options_err); + Stdio.WriteLine(sh, &buf); + break; + default: + break; + } + if (options_err) { + Free(options_err); + Free(ds); + return 1; + } + if (flags & 1 << @shell_cmd_uname_opt_a) + flags = 0x01FF; // Set all options. + for (i = 0; i < 8; i++) { + switch (flags & 1 << i) { + case 1 << @shell_cmd_uname_opt_s: + String.Append(&buf, Define("DD_OS_NAME_VERSION")); + *StrLastOcc(&buf, "V") = NULL; + break; + case 1 << @shell_cmd_uname_opt_n: + String.Append(&buf, "%s ", &sh->session->hostname); + break; + case 1 << @shell_cmd_uname_opt_r: + String.Append(&buf, "%1.2f ", sys_os_version); + break; + case 1 << @shell_cmd_uname_opt_v: + MemCpy(ds_mm, ds_m + ((ds->mon - 1) * 3), 3); + MemCpy(ds_dd, ds_d + (ds->day_of_week * 3), 3); + String.Append(&buf, "%s %s %d %02d:%02d:%02d UTC %d ", ds_dd, ds_mm, + ds->day_of_mon, ds->hour, ds->min, ds->sec, ds->year); + break; + case 1 << @shell_cmd_uname_opt_m: + case 1 << @shell_cmd_uname_opt_p: + case 1 << @shell_cmd_uname_opt_i: + String.Append(&buf, "x86_64 "); + break; + case 1 << @shell_cmd_uname_opt_o: + String.Append(&buf, "Erythros "); + break; + default: + break; + } + } + Stdio.WriteLine(sh, &buf); + Stdio.WriteLine(sh, "\n"); + Free(ds); + return 0; +} \ No newline at end of file diff --git a/System/Shell/Commands/whoami.HC b/System/Shell/Commands/whoami.HC new file mode 100644 index 0000000..5474528 --- /dev/null +++ b/System/Shell/Commands/whoami.HC @@ -0,0 +1,28 @@ +I64 @shell_cmd_whoami(@shell* sh, I64 argc, U8** argv) +{ + U8* options_list = ""; + U64 options_err = NULL; + I64 flags = NULL; + I64 res = 0; + U8 buf[512]; + switch ( + @shell_parse_opts(sh, options_list, argc, argv, &flags, &options_err)) { + case SHELL_OPTS_ERR_INVALID_OPT: + StrPrint(&buf, "uname: unrecognized option -- '%s'\n", options_err); + Stdio.WriteLine(sh, &buf); + res = 1; + break; + case SHELL_OPTS_ERR_EXTRA_OPD: + StrPrint(&buf, "uname: extra operand '%s'\n", options_err); + Stdio.WriteLine(sh, &buf); + res = 1; + break; + default: + Stdio.WriteLine(sh, &sh->session->user.name); + Stdio.WriteLine(sh, "\n"); + break; + } + if (options_err) + Free(options_err); + return res; +} \ No newline at end of file diff --git a/System/Shell/Commands/wpset.HC b/System/Shell/Commands/wpset.HC new file mode 100644 index 0000000..85b46c0 --- /dev/null +++ b/System/Shell/Commands/wpset.HC @@ -0,0 +1,24 @@ +I64 @shell_cmd_wpset(@shell* sh, I64 argc, U8** argv) +{ + U8 buf[512]; + if (argc < 2) { + return 0; + } + I64 size = 0; + U64 fbuf = FileSystem.ReadFile(argv[1], &size); + if (!fbuf) { + StrPrint(&buf, "Error reading file %s\n", argv[1]); + Stdio.WriteLine(sh, &buf); + return 1; + } + Context2D* new = Image.BufferToContext2D(fbuf, size); + Free(fbuf); + if (!new) { + StrPrint(&buf, "Error in Image.BufferToContext2D\n"); + Stdio.WriteLine(sh, &buf); + return 1; + } + Compositor.SetWallpaper(new); + DelContext2D(new); + return 0; +} \ No newline at end of file diff --git a/System/Utilities/Dns.HC b/System/Utilities/Dns.HC new file mode 100644 index 0000000..42c97a7 --- /dev/null +++ b/System/Utilities/Dns.HC @@ -0,0 +1,10 @@ +U0 DnsQuery(U8* host) +{ + U32 result = @dns_query(host); + if (result == U32_MAX) { + "Error looking up host %s\n", host; + return; + } + "Query for %s: %d.%d.%d.%d\n", host, result.u8[3], result.u8[2], result.u8[1], + result.u8[0]; +} diff --git a/System/Utilities/Image.HC b/System/Utilities/Image.HC new file mode 100644 index 0000000..c7383e2 --- /dev/null +++ b/System/Utilities/Image.HC @@ -0,0 +1,414 @@ +Silent(1); // This is needed to suppress "Function should return val" warnings for wrappers to non-HolyC functions + +// class @image +// { +// CDC* (*FromBuffer)(U8* buffer, I64 len); +// CDC* (*Load)(U8* filename); +// CDC* (*Write)(U8* filename, CDC* dc); +// }; +// +// @image Image; + +class @image_frame +{ + CDC* dc; + CSprite* sprite; + I64 delay; +}; + +class @image_collection +{ + @image_frame** frames; + I64 count; + I64 current; + I64 jiffies; + I64 index; + @image_collection* next; +}; + +I64 @image_cbgr24_to_4_bit(CBGR24* ptr, Bool dither_probability) +{ + I64 res, k; + if (dither_probability) { + k = RandU32; + if (SqrI64(ptr->r) + SqrI64(ptr->g) + SqrI64(ptr->b) >= 3 * SqrI64(k.u8[0])) + res = 8; + else + res = 0; + if (ptr->r >= k.u8[1]) + res |= RED; + if (ptr->g >= k.u8[2]) + res |= GREEN; + if (ptr->b >= k.u8[3]) + res |= BLUE; + } else { + if (SqrI64(ptr->r) + SqrI64(ptr->g) + SqrI64(ptr->b) >= SqrI64(0x80)) { + res = 8; + if (ptr->r >= 0x80) + res |= RED; + if (ptr->g >= 0x80) + res |= GREEN; + if (ptr->b >= 0x80) + res |= BLUE; + } else { + res = 0; + if (ptr->r >= 0x40) + res |= RED; + if (ptr->g >= 0x40) + res |= GREEN; + if (ptr->b >= 0x40) + res |= BLUE; + } + } + return res; +} + +#define IMAGE_DITHER_NONE 0 +#define IMAGE_DITHER_NATIVE 1 +#define IMAGE_DITHER_FLOYDSTEINBERG 2 + +U0 @image_render_4bit_floydstein(U8* buffer, I32 width, I32 height) +{ + U64 reg RDI rdi = buffer; + U64 reg RSI rsi = width; + U64 reg RDX rdx = height; + no_warn rdi, rsi, rdx; + asm { + MOV RAX, RENDER_4BIT_FLOYDSTEIN + CALL RAX + } +} + +CDC* @image_render_16color_native(U8* pixels, I32 x, I32 y, Bool dither) +{ + I64 i; + I64 j; + I64 cnt = 0; + CBGR24 cbgr24; + CDC* dc = DCNew(x, y); + for (i = 0; i < y; i++) + for (j = 0; j < x; j++) { + cbgr24.r = pixels[cnt]; + cbgr24.g = pixels[cnt + 1]; + cbgr24.b = pixels[cnt + 2]; + if (!pixels[cnt + 3]) + dc->color = TRANSPARENT; + else + dc->color = @image_cbgr24_to_4_bit(&cbgr24, dither); + GrPlot(dc, j, y - i - 1); + cnt += 4; + } + return dc; +} + +CBGR24 @image_palette_std[COLORS_NUM] = { + 0x000000, 0x0000AA, 0x00AA00, 0x00AAAA, + 0xAA0000, 0xAA00AA, 0xAA5500, 0xAAAAAA, + 0x555555, 0x5555FF, 0x55FF55, 0x55FFFF, + 0xFF5555, 0xFF55FF, 0xFFFF55, 0xFFFFFF +}; + +CBGR24 @image_dif_rgb(CBGR24 from, CBGR24 to) +{ + CBGR24 dif; + dif.r = to.r - from.r; + dif.g = to.g - from.g; + dif.b = to.b - from.b; + return dif; +} + +F64 @image_dist_rgb(CBGR24 from, CBGR24 to) +{ + CBGR24 dif = @image_dif_rgb(from, to); + F64 dist = dif.r * dif.r + dif.g * dif.g + dif.b * dif.b; + return dist; +} + +I64 @image_get_4bit_color(CBGR24* cbgr24) +{ + F64 dist = -1, tempDist; + I64 i; + I64 color = TRANSPARENT; + for (i = 0; i < COLORS_NUM; i++) { + tempDist = @image_dist_rgb(*cbgr24, @image_palette_std[i]); + if (tempDist < dist || dist < 0) { + dist = tempDist; + color = i; + } + } + return color; +} + +CDC* @image_render_16color_floydsteinberg(U8* pixels, I32 width, I32 height) +{ + @image_render_4bit_floydstein(pixels, width, height); + I64 i; + I64 j; + I64 cnt = 0; + CBGR24 cbgr24; + CDC* dc = DCNew(width, height); + for (i = 0; i < height; i++) + for (j = 0; j < width; j++) { + cbgr24.r = pixels[cnt]; + cbgr24.g = pixels[cnt + 1]; + cbgr24.b = pixels[cnt + 2]; + if (!pixels[cnt + 3]) + dc->color = TRANSPARENT; + else + dc->color = @image_get_4bit_color(&cbgr24); + GrPlot(dc, j, height - i - 1); + cnt += 4; + } + return dc; +} + +CDC* @image_generate_dc_from_pixels(U8* pixels, I32 width, I32 height, Bool dither = IMAGE_DITHER_FLOYDSTEINBERG) +{ + switch (dither) { + case IMAGE_DITHER_NONE: + case IMAGE_DITHER_NATIVE: + return @image_render_16color_native(pixels, width, height, dither); + break; + case IMAGE_DITHER_FLOYDSTEINBERG: + return @image_render_16color_floydsteinberg(pixels, width, height); + break; + default: + break; + } + return NULL; +} + +U8* @image_load_gif_from_memory(U8* buffer, I64 len, I64** delays, I64* x, I64* y, + I64* z) +{ + U64 reg RDI rdi = buffer; + U64 reg RSI rsi = len; + U64 reg RDX rdx = delays; + U64 reg RCX rcx = x; + U64 reg R8 r8 = y; + U64 reg R9 r9 = z; + no_warn rdi, rsi, rdx, rcx, r8, r9; + asm { + MOV RAX, IMAGE_LOAD_GIF_FROM_MEMORY + CALL RAX + } +} + +U8* @stbi_failure_reason() +{ + asm { + MOV RAX, STBI_FAILURE_REASON + CALL RAX + } +} + +I32 @stbi_info_from_memory(U8* buffer, I64 len, I64* x, I64* y, I64* comp) +{ + U64 reg RDI rdi = buffer; + U64 reg RSI rsi = len; + U64 reg RDX rdx = x; + U64 reg RCX rcx = y; + U64 reg R8 r8 = comp; + no_warn rdi, rsi, rdx, rcx, r8; + asm { + MOV RAX, STBI_INFO_FROM_MEMORY + CALL RAX + } +} + +U8* @stbi_load_from_memory(U8* buffer, I64 len, I64* x, I64* y, + I64* channels_in_file, I64 desired_channels) +{ + U64 reg RDI rdi = buffer; + U64 reg RSI rsi = len; + U64 reg RDX rdx = x; + U64 reg RCX rcx = y; + U64 reg R8 r8 = channels_in_file; + U64 reg R9 r9 = desired_channels; + no_warn rdi, rsi, rdx, rcx, r8, r9; + asm { + MOV RAX, STBI_LOAD_FROM_MEMORY + CALL RAX + } +} + +U32* @stbi_write_png_to_mem(U32* pixels, I32 stride_bytes, I32 x, I32 y, I32 n, I32* out_len) +{ + U64 reg RDI rdi = pixels; + U64 reg RSI rsi = stride_bytes; + U64 reg RDX rdx = x; + U64 reg RCX rcx = y; + U64 reg R8 r8 = n; + U64 reg R9 r9 = out_len; + no_warn rdi, rsi, rdx, rcx, r8, r9; + asm { + MOV RAX, STBI_WRITE_PNG_TO_MEM + CALL RAX + } +} + +CDC* @image_load(U8* filename) +{ + if (!filename || !FileFind(filename)) { + // PrintErr("Image file not found.\n"); + return NULL; + } + I64 len; + I32 x; + I32 y; + I32 comp; + U8* buffer = FileRead(filename, &len); + I32 code = @stbi_info_from_memory(buffer, len, &x, &y, &comp); + if (code != 1) { + Free(buffer); + return NULL; + } + U8* pixels = @stbi_load_from_memory(buffer, len, &x, &y, &comp, 4); + Free(buffer); + CDC* dc = @image_generate_dc_from_pixels(pixels, x, y); + Free(pixels); + return dc; +} + +U32 @image_rgba_color_table[16] = { + 0xff000000, 0xffaa0000, 0xff00aa00, 0xffaaaa00, + 0xff0000aa, 0xffaa00aa, 0xff0055aa, 0xffaaaaaa, + 0xff555555, 0xffff5555, 0xff55ff55, 0xffffff55, + 0xff5555ff, 0xffff55ff, 0xff55ffff, 0xffffffff +}; + +U32 @image_get_rgba_color(I64 color) +{ + if (color > 15) + return 0; + return @image_rgba_color_table[color]; +} + +U32* @image_get_rgba_buffer_from_dc_body(CDC* dc) +{ + if (!dc) + return NULL; + U32* pixels = CAlloc((dc->width * dc->height) * 4, erythros_mem_task); + I64 x; + I64 y; + I64 p = 0; + for (y = 0; y < dc->height; y++) + for (x = 0; x < dc->width; x++) + pixels[p++] = @image_get_rgba_color(GrPeek(dc, x, y)); + return pixels; +} + +U0 @image_write(U8* filename, CDC* dc) +{ + if (!dc) { + PrintErr("Device context is NULL.\n"); + return; + } + I32 out_len; + U32* rgba_buffer = @image_get_rgba_buffer_from_dc_body(dc); + if (!rgba_buffer) { + PrintErr("RGBA buffer is NULL.\n"); + return; + } + U8* png_buffer = @stbi_write_png_to_mem(rgba_buffer, dc->width * 4, dc->width, dc->height, 4, &out_len); + if (!png_buffer) { + PrintErr("PNG buffer is NULL.\n"); + Free(rgba_buffer); + return; + } + FileWrite(filename, png_buffer, out_len); + Free(rgba_buffer); + Free(png_buffer); +} + +U32 @image_pixel_flip_rgb_bgr(U32 src) +{ + U32 dst; + dst.u8[0] = src.u8[2]; + dst.u8[1] = src.u8[1]; + dst.u8[2] = src.u8[0]; + dst.u8[3] = src.u8[3]; + return dst; +} + +CDC* @image_from_buffer(U8* buffer, I64 len) +{ + I32 x = 0; + I32 y = 0; + U8* pixels = NULL; + CDC* dc = NULL; + + I32 comp; + I32 code = @stbi_info_from_memory(buffer, len, &x, &y, &comp); + if (code != 1) { + return NULL; + } + pixels = @stbi_load_from_memory(buffer, len, &x, &y, &comp, 4); + if (!pixels) + PopUpOk(@stbi_failure_reason); + dc = @image_generate_dc_from_pixels(pixels, x, y); + Free(pixels); + return dc; +} + +@image_collection* @image_collection_from_buffer(U8* buffer, I64 len) +{ + I64 i; + I32* delays; + I32 x; + I32 y; + I32 z; + I32 comp; + I32 code = @stbi_info_from_memory(buffer, len, &x, &y, &comp); + if (code != 1) { + return NULL; + } + U64 pixels = @image_load_gif_from_memory(buffer, len, &delays, &x, &y, &z); + if (!pixels) + PopUpOk(@stbi_failure_reason); + if (!z) + return NULL; // no frames? + @image_collection* collection = CAlloc(sizeof(@image_collection), erythros_mem_task); + @image_frame* frame; + collection->frames = CAlloc(sizeof(@image_frame*) * z, erythros_mem_task); + collection->count = z; + for (i = 0; i < z; i++) { + frame = CAlloc(sizeof(@image_frame), erythros_mem_task); + frame->dc = @image_generate_dc_from_pixels(pixels, x, y); + frame->sprite = DC2Sprite(frame->dc); + frame->delay = delays[i]; + collection->frames[i] = frame; + pixels += (x * y) * 4; + } + return collection; +} + +// Image.FromBuffer = &@image_from_buffer; +// Image.Load = &@image_load; +// Image.Write = &@image_write; + +Silent(0); + +U0 Screenshot(U8* custom_filename = NULL, Bool output_filename_to_focus_task = FALSE) +{ + CDC* dc = DCScrnCapture; + U8 filename[256]; + CDateStruct ds; + if (custom_filename) + StrCpy(filename, custom_filename); + else { + Date2Struct(&ds, Now); + StrPrint(filename, "C:/Tmp/ScrnShots/%04d-%02d-%02d-%02d-%02d-%02d.png", ds.year, ds.mon, ds.day_of_mon, ds.hour, ds.min, ds.sec); + } + @image_write(filename, dc); + DCDel(dc); + if (output_filename_to_focus_task) + XTalk(sys_focus_task, filename); +}; + +U0 @screenshot_hotkey(I64) +{ + Screenshot("C:/Home/Screenshot.png", TRUE); +} + +CtrlAltCBSet('S', &@screenshot_hotkey, "", , FALSE); diff --git a/System/Utilities/NetRep.HC b/System/Utilities/NetRep.HC new file mode 100644 index 0000000..f56bacd --- /dev/null +++ b/System/Utilities/NetRep.HC @@ -0,0 +1,22 @@ +U0 NetRep() +{ + NetInfoRequest* req = @net_info_request; + "MAC address : %02x:%02x:%02x:%02x:%02x:%02x\n", req->mac_address.u8[5], req->mac_address.u8[4], + req->mac_address.u8[3], req->mac_address.u8[2], + req->mac_address.u8[1], req->mac_address.u8[0]; + "IPv4 address : %d.%d.%d.%d\n", req->ipv4_address.u8[3], req->ipv4_address.u8[2], + req->ipv4_address.u8[1], req->ipv4_address.u8[0]; + "IPv4 netmask : %d.%d.%d.%d\n", req->ipv4_netmask.u8[3], req->ipv4_netmask.u8[2], + req->ipv4_netmask.u8[1], req->ipv4_netmask.u8[0]; + "IPv4 network : %d.%d.%d.%d\n", req->ipv4_network.u8[3], req->ipv4_network.u8[2], + req->ipv4_network.u8[1], req->ipv4_network.u8[0]; + "IPv4 gateway : %d.%d.%d.%d\n", req->ipv4_gateway.u8[3], req->ipv4_gateway.u8[2], + req->ipv4_gateway.u8[1], req->ipv4_gateway.u8[0]; + "DNS server (port) : %d.%d.%d.%d (%d)\n", req->dns_server_address.u8[3], req->dns_server_address.u8[2], + req->dns_server_address.u8[1], req->dns_server_address.u8[0], req->dns_server_port; + "RX bytes : %d\n", req->rx_bytes; + "RX frames : %d\n", req->rx_frames; + "TX bytes : %d\n", req->tx_bytes; + "TX frames : %d\n", req->tx_frames; + Free(req); +} diff --git a/System/Utilities/Ping.HC b/System/Utilities/Ping.HC new file mode 100644 index 0000000..386c441 --- /dev/null +++ b/System/Utilities/Ping.HC @@ -0,0 +1,72 @@ +#define PING_ERR_INVALID_HOST 1 +#define PING_ERR_HOST_NOT_FOUND 2 + +#define PING_PAYLOAD_SIZE 56 + +I64 @ping_err(I64 code) +{ + switch (code) { + case PING_ERR_INVALID_HOST: + "Invalid host specified\n"; + return 1; + break; + case PING_ERR_HOST_NOT_FOUND: + "Host not found\n"; + return 2; + break; + default: + "Unspecified error\n"; + return -1; + } +} + +I64 Ping(U8* host, I64 count = 4) +{ + if (!host) + return @ping_err(PING_ERR_INVALID_HOST); + if (!StrLen(host)) + return @ping_err(PING_ERR_INVALID_HOST); + + U32 addr = @dns_query(host); + if (addr == U32_MAX) + return @ping_err(PING_ERR_HOST_NOT_FOUND); + + U16 iden = (RandU16 * SysTimerRead) & 0xFFFF; + I64 start_jiffies; + U32 reply = NULL; + I64 res = 0; + U16 seq = 0; + I64 loss = 0; + + IcmpRequest* request = CAlloc(sizeof(IcmpRequest), Fs->code_heap); + + "PING %s (%d.%d.%d.%d): %d data bytes\n", + host, addr.u8[3], addr.u8[2], addr.u8[1], addr.u8[0], PING_PAYLOAD_SIZE; + + I64 i; + for (i = 0; i < count; i++) { + start_jiffies = cnts.jiffies; + reply = @icmp_echo_request(addr, iden, seq, request, i); + if (!reply) { + "Request timeout for icmp_seq %d\n", seq; + ++loss; + res = 1; + } else { + "%d bytes from %d.%d.%d.%d: icmp_seq=%d ttl=%d time=%d ms\n", + reply.u16[1], addr.u8[3], addr.u8[2], addr.u8[1], addr.u8[0], seq, reply.u16[0], cnts.jiffies - start_jiffies; + } + while (cnts.jiffies < start_jiffies + 1000 && i < (count - 1)) + Sleep(1); + ++seq; + } + + Free(request); + + "--- %d.%d.%d.%d ping statistics ---\n", addr.u8[3], addr.u8[2], addr.u8[1], addr.u8[0]; + "%d packets transmitted, %d packets received, %0f", + seq, seq - loss, (loss * 1.0 / seq * 1.0) * 100; + PutChars(37); + " packet loss\n"; + + return res; +} \ No newline at end of file diff --git a/System/Utilities/Time.HC b/System/Utilities/Time.HC new file mode 100644 index 0000000..8025c76 --- /dev/null +++ b/System/Utilities/Time.HC @@ -0,0 +1,92 @@ +U0 @time_cmos_update_byte(I64 time_reg, I64 val) +{ + OutU8(0x70, time_reg); + OutU8(0x71, val); +} + +I64 @time_dec_to_bcd(I64 val) +{ + return (((val / 10) << 4) | (val % 10)); +} + +U0 @time_update(U8* date_str, I64 mS_delta, I64 hour_offset) +{ + no_warn mS_delta; + Bool is_bcd; + OutU8(0x70, 0x0B); + if (InU8(0x71) & 4) + is_bcd = FALSE; + else + is_bcd = TRUE; + + I64 date_argc; + U8** date_argv = String.Split(date_str, ' ', &date_argc); + + I64 month = DefineMatch(date_argv[2], "ST_MONTHS") + 1; + I64 day = Str2I64(date_argv[1]); + I64 year = Str2I64(date_argv[3] + 2); + I64 century = 20; + + date_argv[4][2] = NULL; + date_argv[4][5] = NULL; + + I64 hour = Str2I64(date_argv[4]); + I64 minute = Str2I64(date_argv[4] + 3); + I64 second = Str2I64(date_argv[4] + 6); + + // FIXME: Handle month boundaries, and 12 hour time + hour += hour_offset; + if (hour < 0) { + hour += 24; + --day; + } else if (hour > 23) { + hour -= 24; + ++day; + } + + if (is_bcd) { + century = @time_dec_to_bcd(century); + year = @time_dec_to_bcd(year); + month = @time_dec_to_bcd(month); + day = @time_dec_to_bcd(day); + hour = @time_dec_to_bcd(hour); + minute = @time_dec_to_bcd(minute); + second = @time_dec_to_bcd(second); + } + + @time_cmos_update_byte(0x32, century); + @time_cmos_update_byte(0x09, year); + @time_cmos_update_byte(0x08, month); + @time_cmos_update_byte(0x07, day); + @time_cmos_update_byte(0x04, hour); + @time_cmos_update_byte(0x02, minute); + @time_cmos_update_byte(0x00, second); +} + +I64 @time_tz_offset() +{ + return -4; +} + +U0 @time_query(Bool set = FALSE) +{ + U8 buf[1024]; + @http_url* url = @http_parse_url("http://time.google.com"); + @http_response* resp = Http.Head(url, &buf); + while (resp->state != HTTP_STATE_DONE) + Sleep(1); + I64 mS_delta = cnts.jiffies; + "Set current date and time to %s ", resp->headers->@("Date"); + if (!set) + set = YorN; + else + "\n"; + if (set) + @time_update(resp->headers->@("Date"), mS_delta, @time_tz_offset); +} + +U0 TimeSync() +{ + Sleep(500); + @time_query(1); +} diff --git a/scripts/build-all b/scripts/build-all new file mode 100755 index 0000000..1b99bfb --- /dev/null +++ b/scripts/build-all @@ -0,0 +1,191 @@ +#!/usr/bin/python3 +from pathlib import Path +import glob +import os +import subprocess +import sys +import time + +if len(sys.argv) < 2: + raise ValueError('wrong number of arguments') + +project_path = sys.argv[1] + '/' +project_name = project_path.rsplit('/')[-2] + +isoc_file = project_path + 'build/isoc/Erythros.ISO.C' +redsea_path = project_path + 'build/redsea' + +home_path = str(Path.home()) + '/' + +jakt_compiler_path = home_path + 'cloned/jakt/build/bin/jakt' +jakt_runtime_path = home_path + 'cloned/jakt/runtime' +jakt_lib_path = home_path + 'cloned/jakt/build/lib/x86_64-unknown-linux-unknown/' + +qemu_slipstream_iso_file = project_path + 'build/isoc/bootable.iso' +qemu_virtio_disk_path = home_path + 'erythros-virtio-disk.qcow2' + +qemu_bin_path = "qemu-system-x86_64" +qemu_display = "-display sdl,grab-mod=rctrl" + +templeos_iso_file = home_path + 'iso/TempleOS.ISO' + +qemu_run_cmd = qemu_bin_path + ' ' + qemu_display + ' -enable-kvm -smp cores=4 -m 8192 -netdev tap,id=mynet0,ifname=tap0,script=no,downscript=no -device ac97 -device virtio-net,netdev=mynet0 -drive file=' + qemu_virtio_disk_path + ',format=qcow2,if=none,index=0,media=disk,id=virtio-disk -device virtio-blk-pci,drive=virtio-disk -device vmmouse,i8042=i8042 -device vmware-svga -cdrom ' + qemu_slipstream_iso_file + ' -debugcon stdio -boot d' + +def clang_format_src_files(): + print("build-all: clang-format-src-files") + exclude_paths = ["stb_", "tlse", ".iso.c"] + format_file_extensions = [".c", ".cpp", ".h", ".hc"] + for src_file in glob.glob(project_path + "**", recursive=True): + exclude_file = False + for exclude_path in exclude_paths: + if src_file.lower().find(exclude_path) > 0: + exclude_file = True + if exclude_file: + continue + for format_file_extension in format_file_extensions: + if src_file.lower().endswith(format_file_extension): + print(src_file) + res = os.system('clang-format -i --style=file:' + project_path + '.clang-format ' + src_file) + if res: + raise ValueError("build-all: step 'clang-format-src-files' failed, error code " + str(res)) + +def refresh_build_path(): + print("build-all: refresh-build-path") + res = os.system('rm -rf ' + project_path + 'build && mkdir -p ' + project_path + 'build/bin && mkdir -p ' + project_path + 'build/isoc && mkdir -p ' + project_path + 'build/lib && mkdir -p ' + project_path + 'build/redsea') + if res: + raise ValueError("build-all: step 'refresh-build-path' failed, error code " + str(res)) + +def build_image(): + print("build-all: build-image") + build_specific_options = '-Wl,--section-start=.text=0x1004000 -Wl,--section-start=.plt=0x1002020 -no-pie' + res = os.system('cd ' + project_path + '&& cd src/image && gcc -o ../../build/bin/image ' + build_specific_options + ' -O0 -mno-mmx -mno-red-zone image.c') + if res: + raise ValueError("build-all: step 'build-image' failed, error code " + str(res)) + +def build_libtemple(): + print("build-all: build-libtemple") + res = os.system('cd ' + project_path + 'src/libtemple && g++ -c -o ../../build/libtemple.o libtemple.cpp && gcc -shared -o ../../build/lib/libtemple.so ../../build/libtemple.o && rm ' + project_path + 'build/libtemple.o') + if res: + raise ValueError("build-all: step 'build-libtemple' failed, error code " + str(res)) + +def build_tlse(): + print("build-all: build-tlse") + build_specific_options = '-Wl,--section-start=.text=0x1204000 -Wl,--section-start=.plt=0x1202020 -no-pie' + res = os.system('cd ' + project_path + '&& cd src/tlse && gcc -o ../../build/bin/tlse ' + build_specific_options + ' -O0 -mno-mmx -mno-red-zone -DTLS_AMALGAMATION tlse.c') + if res: + raise ValueError("build-all: step 'build-tlse' failed, error code " + str(res)) + +def transpile_net_to_sepples(): + print("build-all: transpile-net-to-sepples") + res = os.system('cd ' + project_path + 'src/net && ' + jakt_compiler_path + ' -S -R ' + jakt_runtime_path + ' -B ' + project_path + 'build/net -O net.jakt') + if res: + raise ValueError("build-all: step 'transpile-net-to-sepples' failed, error code " + str(res)) + +def build_net(): + print("build-all: build-net") + build_specific_options = '-Wno-invalid-offsetof -Wl,--section-start=.text=0x1404000 -Wl,--section-start=.plt=0x1402020 -no-pie' + res = os.system('cd ' + project_path + 'build/net && clang++-19 ' + build_specific_options + ' -O3 -I ' + jakt_runtime_path + ' -I ' + project_path + '/src/libtemple -fcolor-diagnostics -std=c++20 -fno-exceptions -Wno-user-defined-literals -Wno-deprecated-declarations -Wno-parentheses-equality -Wno-unqualified-std-cast-call -Wno-unknown-warning-option -Wno-int-to-pointer-cast -mno-red-zone -o ../bin/net *.cpp ../lib/libtemple.so ' + jakt_lib_path + 'libjakt_runtime_x86_64-unknown-linux-unknown.a ' + jakt_lib_path + 'libjakt_main_x86_64-unknown-linux-unknown.a && cd .. && rm -rf net') + if res: + raise ValueError("build-all: step 'build-net' failed, error code " + str(res)) + +def address_string_for_symbol(file, symbol): + p = subprocess.Popen('readelf -s --wide "' + file + '" | grep \'' + symbol + '$\' | awk \'{sub("000000000", "0x", $2); print $2}\'', shell=True, stdin=subprocess.PIPE, stdout=subprocess.PIPE) + return str(p.communicate()[0][:-1].decode(encoding='utf-8')) + +def image_hc_fixup(macro, symbol, image_bin_path, image_hc_path): + os.system('echo -e "#define ' + macro + ' ' + address_string_for_symbol(image_bin_path, symbol) + '\n" | cat - ' + image_hc_path + ' | sponge ' + image_hc_path) + return + +def tlse_hc_fixup(macro, symbol, tlse_bin_path, tlse_hc_path): + os.system('echo -e "#define ' + macro + ' ' + address_string_for_symbol(tlse_bin_path, symbol) + '\n" | cat - ' + tlse_hc_path + ' | sponge ' + tlse_hc_path) + return + +def generate_iso_c_file(): + print("build-all: generate-iso-c-file") + step_error_message = "build-all: step 'generate-iso-c-file' failed, error code " + + try: + os.remove(isoc_file) + except: + pass + res = os.system('isoc-mount --rw ' + isoc_file + ' ' + redsea_path) + if res: + raise ValueError(step_error_message + str(res)) + time.sleep(0.25) + + copy_files_cmd_line = 'rsync -av --inplace --progress ' + project_path + ' ' + redsea_path + copy_files_cmd_line += ' --exclude .clang-format' + copy_files_cmd_line += ' --exclude .git' + copy_files_cmd_line += ' --exclude .gitignore' + copy_files_cmd_line += ' --exclude .vscode' + copy_files_cmd_line += ' --exclude build/isoc' + copy_files_cmd_line += ' --exclude build/lib' + copy_files_cmd_line += ' --exclude build/redsea' + copy_files_cmd_line += ' --exclude scripts' + copy_files_cmd_line += ' --exclude src' + res = os.system(copy_files_cmd_line) + if res: + raise ValueError(step_error_message + str(res)) + + # Fixup addresses for Image.HC + image_bin_path = redsea_path + '/build/bin/image' + image_hc_path = redsea_path + '/System/Utilities/Image.HC' + + image_hc_fixup('IMAGE_LOAD_GIF_FROM_MEMORY', 'image_load_gif_from_memory', image_bin_path, image_hc_path) + image_hc_fixup('STBI_WRITE_PNG_TO_MEM', 'stbi_write_png_to_mem', image_bin_path, image_hc_path) + image_hc_fixup('STBI_LOAD_FROM_MEMORY', 'stbi_load_from_memory', image_bin_path, image_hc_path) + image_hc_fixup('STBI_INFO_FROM_MEMORY', 'stbi_info_from_memory', image_bin_path, image_hc_path) + image_hc_fixup('STBI_FAILURE_REASON', 'stbi_failure_reason', image_bin_path, image_hc_path) + image_hc_fixup('RENDER_4BIT_FLOYDSTEIN', 'render_4bit_floydstein', image_bin_path, image_hc_path) + + # Fixup addresses for Tlse.HC + + rsa_hc_path = redsea_path + '/System/Libraries/Rsa.HC' + tlse_bin_path = redsea_path + '/build/bin/tlse' + tlse_hc_path = redsea_path + '/System/Libraries/Tlse.HC' + + tlse_hc_fixup('RSA_IMPORT', 'rsa_import', tlse_bin_path, rsa_hc_path) + tlse_hc_fixup('RSA_CREATE_SIGNATURE', 'rsa_create_signature', tlse_bin_path, rsa_hc_path) + tlse_hc_fixup('RSA_VERIFY_SIGNATURE', 'rsa_verify_signature', tlse_bin_path, rsa_hc_path) + tlse_hc_fixup('TLS_CREATE_CONTEXT', 'tls_create_context', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_SNI_SET', 'tls_sni_set', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_CLIENT_CONNECT', 'tls_client_connect', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_CONNECTION_STATUS', 'tls_connection_status', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_GET_WRITE_BUFFER', 'tls_get_write_buffer', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_BUFFER_CLEAR', 'tls_buffer_clear', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_CONSUME_STREAM', 'tls_consume_stream', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_READ', 'tls_read', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_WRITE', 'tls_write', tlse_bin_path, tlse_hc_path) + tlse_hc_fixup('TLS_ESTABLISHED', 'tls_established', tlse_bin_path, tlse_hc_path) + time.sleep(0.25) + + res = os.system('sync && fusermount -u ' + redsea_path) + if res: + raise ValueError(step_error_message + str(res)) + time.sleep(0.25) + +def generate_slipstream_iso_file(): + print("build-all: generate-slipstream-iso-file") + res = os.system('templeos-slipstream ' + templeos_iso_file + ' ' + isoc_file + ' ' + qemu_slipstream_iso_file) + if res: + raise ValueError("build-all: step 'generate-slipstream-iso-file' failed, error code " + str(res)) + +def run(): + print("build-all: run") + res = os.system(qemu_run_cmd) + if res: + raise ValueError("build-all: step 'run' failed, error code " + str(res)) + +def build_all(): + clang_format_src_files() + refresh_build_path() + build_image() + build_libtemple() + build_tlse() + transpile_net_to_sepples() + build_net() + generate_iso_c_file() + generate_slipstream_iso_file() + run() + +build_all() \ No newline at end of file diff --git a/src/image/image.c b/src/image/image.c new file mode 100644 index 0000000..42f031c --- /dev/null +++ b/src/image/image.c @@ -0,0 +1,209 @@ +#define STBI_WRITE_NO_STDIO +#define STB_IMAGE_WRITE_STATIC +#define STB_IMAGE_WRITE_IMPLEMENTATION +#define STB_IMAGE_IMPLEMENTATION +#define STBI_NO_LINEAR +#define STBI_NO_STDIO +#define STBI_NO_SIMD +#define STBI_NO_HDR + +#include "stb_image.h" +#include "stb_image_write.h" + +int main() { return 0; } + +STBIDEF stbi_uc* image_load_gif_from_memory(stbi_uc const* buffer, int len, + int** delays, int* x, int* y, + int* z) +{ + int comp; + return stbi_load_gif_from_memory(buffer, len, delays, x, y, z, &comp, 4); +} + +/* dither.c: MIT License + +Copyright (c) 2016 jonmortiboy + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + +*/ + +typedef struct RGB { + int r; + int g; + int b; +} RGB; + +int imgw, imgh; + +// Define the 4bit colour palette +int numCols = 16; +RGB cols4bit[] = { + { 0, 0, 0 }, { 0, 0, 170 }, { 0, 170, 0 }, { 0, 170, 170 }, + { 170, 0, 0 }, { 170, 0, 170 }, { 170, 85, 0 }, { 170, 170, 170 }, + { 85, 85, 85 }, { 85, 85, 255 }, { 85, 255, 85 }, { 85, 255, 255 }, + { 255, 85, 85 }, { 255, 85, 255 }, { 255, 255, 85 }, { 255, 255, 255 } +}; +RGB* cols = cols4bit; + +RGB getRGB(uint32_t* pixels, int x, int y); +void setRGB(uint32_t* pixels, int x, int y, RGB rgb); +RGB difRGB(RGB from, RGB to); +RGB addRGB(RGB a, RGB b); +RGB divRGB(RGB rgb, double d); +RGB mulRGB(RGB rgb, double d); +RGB nearestRGB(RGB rgb, RGB* rgbs, int numRGBs); +double distRGB(RGB from, RGB to); + +void render_4bit_floydstein(uint32_t* pixels, int width, int height); + +RGB getRGB(uint32_t* pixels, int x, int y) +{ + RGB rgb; + rgb.r = 0; + rgb.g = 0; + rgb.b = 0; + + if (x < 0 || x >= imgw || y < 0 || y >= imgh) + return rgb; + + rgb.r = (pixels[y * imgw + x] & 0xff); + rgb.g = (pixels[y * imgw + x] & 0xff00) >> 8; + rgb.b = (pixels[y * imgw + x] & 0xff0000) >> 16; + + return rgb; +} + +void setRGB(uint32_t* pixels, int x, int y, RGB rgb) +{ + if (x < 0 || x >= imgw || y < 0 || y >= imgh) + return; + + uint32_t alpha = pixels[y * imgw + x] & 0xff000000; + pixels[y * imgw + x] = alpha + (rgb.r) + (rgb.g << 8) + (rgb.b << 16); +} + +RGB difRGB(RGB from, RGB to) +{ + RGB dif; + dif.r = to.r - from.r; + dif.g = to.g - from.g; + dif.b = to.b - from.b; + + return dif; +} + +RGB addRGB(RGB a, RGB b) +{ + RGB sum; + sum.r = a.r + b.r; + sum.g = a.g + b.g; + sum.b = a.b + b.b; + + if (sum.r > 255) + sum.r = 255; + if (sum.r < 0) + sum.r = 0; + if (sum.g > 255) + sum.g = 255; + if (sum.g < 0) + sum.g = 0; + if (sum.b > 255) + sum.b = 255; + if (sum.b < 0) + sum.b = 0; + + return sum; +} + +RGB divRGB(RGB rgb, double d) +{ + RGB div; + div.r = (int)((double)rgb.r / d); + div.g = (int)((double)rgb.g / d); + div.b = (int)((double)rgb.b / d); + + return div; +} + +RGB mulRGB(RGB rgb, double d) +{ + RGB mul; + mul.r = (int)((double)rgb.r * d); + mul.g = (int)((double)rgb.g * d); + mul.b = (int)((double)rgb.b * d); + + return mul; +} + +double distRGB(RGB from, RGB to) +{ + RGB dif = difRGB(from, to); + double dist = dif.r * dif.r + dif.g * dif.g + dif.b * dif.b; + + return dist; +} + +RGB nearestRGB(RGB rgb, RGB rgbs[], int numRGBs) +{ + double dist = -1, tempDist; + RGB nearest; + + int i; + for (i = 0; i < numRGBs; i++) { + tempDist = distRGB(rgb, rgbs[i]); + + if (tempDist < dist || dist < 0) { + dist = tempDist; + nearest = rgbs[i]; + } + } + + return nearest; +} + +void render_4bit_floydstein(uint32_t* pixels, int width, int height) +{ + + int i, x, y; + imgw = width; + imgh = height; + RGB rgb, nearest, rgberror; + for (i = 0; i < imgw * imgh; i++) { + rgb = getRGB(pixels, i % imgw, i / imgw); + nearest = nearestRGB(rgb, cols, numCols); + + rgberror = difRGB(nearest, rgb); + rgberror = divRGB(rgberror, 16); + + x = i % imgw; + y = i / imgw; + + setRGB(pixels, x + 1, y, + addRGB(getRGB(pixels, x + 1, y), mulRGB(rgberror, 7))); + setRGB(pixels, x - 1, y + 1, + addRGB(getRGB(pixels, x - 1, y + 1), mulRGB(rgberror, 3))); + setRGB(pixels, x, y + 1, + addRGB(getRGB(pixels, x, y + 1), mulRGB(rgberror, 5))); + setRGB(pixels, x + 1, y + 1, + addRGB(getRGB(pixels, x + 1, y + 1), rgberror)); + + setRGB(pixels, i % imgw, i / imgw, nearest); + } +} \ No newline at end of file diff --git a/src/image/stb_image.h b/src/image/stb_image.h new file mode 100644 index 0000000..d152aa9 --- /dev/null +++ b/src/image/stb_image.h @@ -0,0 +1,8634 @@ +/* stb_image - v2.30 - public domain image loader - http://nothings.org/stb + no warranty implied; use at your own risk + + Do this: + #define STB_IMAGE_IMPLEMENTATION + before you include this file in *one* C or C++ file to create the implementation. + + // i.e. it should look like this: + #include ... + #include ... + #include ... + #define STB_IMAGE_IMPLEMENTATION + #include "stb_image.h" + + You can #define STBI_ASSERT(x) before the #include to avoid using assert.h. + And #define STBI_MALLOC, STBI_REALLOC, and STBI_FREE to avoid using malloc,realloc,free + + + QUICK NOTES: + Primarily of interest to game developers and other people who can + avoid problematic images and only need the trivial interface + + JPEG baseline & progressive (12 bpc/arithmetic not supported, same as stock IJG lib) + PNG 1/2/4/8/16-bit-per-channel + + TGA (not sure what subset, if a subset) + BMP non-1bpp, non-RLE + PSD (composited view only, no extra channels, 8/16 bit-per-channel) + + GIF (*comp always reports as 4-channel) + HDR (radiance rgbE format) + PIC (Softimage PIC) + PNM (PPM and PGM binary only) + + Animated GIF still needs a proper API, but here's one way to do it: + http://gist.github.com/urraka/685d9a6340b26b830d49 + + - decode from memory or through FILE (define STBI_NO_STDIO to remove code) + - decode from arbitrary I/O callbacks + - SIMD acceleration on x86/x64 (SSE2) and ARM (NEON) + + Full documentation under "DOCUMENTATION" below. + + +LICENSE + + See end of file for license information. + +RECENT REVISION HISTORY: + + 2.30 (2024-05-31) avoid erroneous gcc warning + 2.29 (2023-05-xx) optimizations + 2.28 (2023-01-29) many error fixes, security errors, just tons of stuff + 2.27 (2021-07-11) document stbi_info better, 16-bit PNM support, bug fixes + 2.26 (2020-07-13) many minor fixes + 2.25 (2020-02-02) fix warnings + 2.24 (2020-02-02) fix warnings; thread-local failure_reason and flip_vertically + 2.23 (2019-08-11) fix clang static analysis warning + 2.22 (2019-03-04) gif fixes, fix warnings + 2.21 (2019-02-25) fix typo in comment + 2.20 (2019-02-07) support utf8 filenames in Windows; fix warnings and platform ifdefs + 2.19 (2018-02-11) fix warning + 2.18 (2018-01-30) fix warnings + 2.17 (2018-01-29) bugfix, 1-bit BMP, 16-bitness query, fix warnings + 2.16 (2017-07-23) all functions have 16-bit variants; optimizations; bugfixes + 2.15 (2017-03-18) fix png-1,2,4; all Imagenet JPGs; no runtime SSE detection on GCC + 2.14 (2017-03-03) remove deprecated STBI_JPEG_OLD; fixes for Imagenet JPGs + 2.13 (2016-12-04) experimental 16-bit API, only for PNG so far; fixes + 2.12 (2016-04-02) fix typo in 2.11 PSD fix that caused crashes + 2.11 (2016-04-02) 16-bit PNGS; enable SSE2 in non-gcc x64 + RGB-format JPEG; remove white matting in PSD; + allocate large structures on the stack; + correct channel count for PNG & BMP + 2.10 (2016-01-22) avoid warning introduced in 2.09 + 2.09 (2016-01-16) 16-bit TGA; comments in PNM files; STBI_REALLOC_SIZED + + See end of file for full revision history. + + + ============================ Contributors ========================= + + Image formats Extensions, features + Sean Barrett (jpeg, png, bmp) Jetro Lauha (stbi_info) + Nicolas Schulz (hdr, psd) Martin "SpartanJ" Golini (stbi_info) + Jonathan Dummer (tga) James "moose2000" Brown (iPhone PNG) + Jean-Marc Lienher (gif) Ben "Disch" Wenger (io callbacks) + Tom Seddon (pic) Omar Cornut (1/2/4-bit PNG) + Thatcher Ulrich (psd) Nicolas Guillemot (vertical flip) + Ken Miller (pgm, ppm) Richard Mitton (16-bit PSD) + github:urraka (animated gif) Junggon Kim (PNM comments) + Christopher Forseth (animated gif) Daniel Gibson (16-bit TGA) + socks-the-fox (16-bit PNG) + Jeremy Sawicki (handle all ImageNet JPGs) + Optimizations & bugfixes Mikhail Morozov (1-bit BMP) + Fabian "ryg" Giesen Anael Seghezzi (is-16-bit query) + Arseny Kapoulkine Simon Breuss (16-bit PNM) + John-Mark Allen + Carmelo J Fdez-Aguera + + Bug & warning fixes + Marc LeBlanc David Woo Guillaume George Martins Mozeiko + Christpher Lloyd Jerry Jansson Joseph Thomson Blazej Dariusz Roszkowski + Phil Jordan Dave Moore Roy Eltham + Hayaki Saito Nathan Reed Won Chun + Luke Graham Johan Duparc Nick Verigakis the Horde3D community + Thomas Ruf Ronny Chevalier github:rlyeh + Janez Zemva John Bartholomew Michal Cichon github:romigrou + Jonathan Blow Ken Hamada Tero Hanninen github:svdijk + Eugene Golushkov Laurent Gomila Cort Stratton github:snagar + Aruelien Pocheville Sergio Gonzalez Thibault Reuille github:Zelex + Cass Everitt Ryamond Barbiero github:grim210 + Paul Du Bois Engin Manap Aldo Culquicondor github:sammyhw + Philipp Wiesemann Dale Weiler Oriol Ferrer Mesia github:phprus + Josh Tobin Neil Bickford Matthew Gregan github:poppolopoppo + Julian Raschke Gregory Mullen Christian Floisand github:darealshinji + Baldur Karlsson Kevin Schmidt JR Smith github:Michaelangel007 + Brad Weinberger Matvey Cherevko github:mosra + Luca Sas Alexander Veselov Zack Middleton [reserved] + Ryan C. Gordon [reserved] [reserved] + DO NOT ADD YOUR NAME HERE + + Jacko Dirks + + To add your name to the credits, pick a random blank space in the middle and fill it. + 80% of merge conflicts on stb PRs are due to people adding their name at the end + of the credits. +*/ + +#ifndef STBI_INCLUDE_STB_IMAGE_H +# define STBI_INCLUDE_STB_IMAGE_H + +// DOCUMENTATION +// +// Limitations: +// - no 12-bit-per-channel JPEG +// - no JPEGs with arithmetic coding +// - GIF always returns *comp=4 +// +// Basic usage (see HDR discussion below for HDR usage): +// int x,y,n; +// unsigned char *data = stbi_load(filename, &x, &y, &n, 0); +// // ... process data if not NULL ... +// // ... x = width, y = height, n = # 8-bit components per pixel ... +// // ... replace '0' with '1'..'4' to force that many components per pixel +// // ... but 'n' will always be the number that it would have been if you said 0 +// stbi_image_free(data); +// +// Standard parameters: +// int *x -- outputs image width in pixels +// int *y -- outputs image height in pixels +// int *channels_in_file -- outputs # of image components in image file +// int desired_channels -- if non-zero, # of image components requested in result +// +// The return value from an image loader is an 'unsigned char *' which points +// to the pixel data, or NULL on an allocation failure or if the image is +// corrupt or invalid. The pixel data consists of *y scanlines of *x pixels, +// with each pixel consisting of N interleaved 8-bit components; the first +// pixel pointed to is top-left-most in the image. There is no padding between +// image scanlines or between pixels, regardless of format. The number of +// components N is 'desired_channels' if desired_channels is non-zero, or +// *channels_in_file otherwise. If desired_channels is non-zero, +// *channels_in_file has the number of components that _would_ have been +// output otherwise. E.g. if you set desired_channels to 4, you will always +// get RGBA output, but you can check *channels_in_file to see if it's trivially +// opaque because e.g. there were only 3 channels in the source image. +// +// An output image with N components has the following components interleaved +// in this order in each pixel: +// +// N=#comp components +// 1 grey +// 2 grey, alpha +// 3 red, green, blue +// 4 red, green, blue, alpha +// +// If image loading fails for any reason, the return value will be NULL, +// and *x, *y, *channels_in_file will be unchanged. The function +// stbi_failure_reason() can be queried for an extremely brief, end-user +// unfriendly explanation of why the load failed. Define STBI_NO_FAILURE_STRINGS +// to avoid compiling these strings at all, and STBI_FAILURE_USERMSG to get slightly +// more user-friendly ones. +// +// Paletted PNG, BMP, GIF, and PIC images are automatically depalettized. +// +// To query the width, height and component count of an image without having to +// decode the full file, you can use the stbi_info family of functions: +// +// int x,y,n,ok; +// ok = stbi_info(filename, &x, &y, &n); +// // returns ok=1 and sets x, y, n if image is a supported format, +// // 0 otherwise. +// +// Note that stb_image pervasively uses ints in its public API for sizes, +// including sizes of memory buffers. This is now part of the API and thus +// hard to change without causing breakage. As a result, the various image +// loaders all have certain limits on image size; these differ somewhat +// by format but generally boil down to either just under 2GB or just under +// 1GB. When the decoded image would be larger than this, stb_image decoding +// will fail. +// +// Additionally, stb_image will reject image files that have any of their +// dimensions set to a larger value than the configurable STBI_MAX_DIMENSIONS, +// which defaults to 2**24 = 16777216 pixels. Due to the above memory limit, +// the only way to have an image with such dimensions load correctly +// is for it to have a rather extreme aspect ratio. Either way, the +// assumption here is that such larger images are likely to be malformed +// or malicious. If you do need to load an image with individual dimensions +// larger than that, and it still fits in the overall size limit, you can +// #define STBI_MAX_DIMENSIONS on your own to be something larger. +// +// =========================================================================== +// +// UNICODE: +// +// If compiling for Windows and you wish to use Unicode filenames, compile +// with +// #define STBI_WINDOWS_UTF8 +// and pass utf8-encoded filenames. Call stbi_convert_wchar_to_utf8 to convert +// Windows wchar_t filenames to utf8. +// +// =========================================================================== +// +// Philosophy +// +// stb libraries are designed with the following priorities: +// +// 1. easy to use +// 2. easy to maintain +// 3. good performance +// +// Sometimes I let "good performance" creep up in priority over "easy to maintain", +// and for best performance I may provide less-easy-to-use APIs that give higher +// performance, in addition to the easy-to-use ones. Nevertheless, it's important +// to keep in mind that from the standpoint of you, a client of this library, +// all you care about is #1 and #3, and stb libraries DO NOT emphasize #3 above all. +// +// Some secondary priorities arise directly from the first two, some of which +// provide more explicit reasons why performance can't be emphasized. +// +// - Portable ("ease of use") +// - Small source code footprint ("easy to maintain") +// - No dependencies ("ease of use") +// +// =========================================================================== +// +// I/O callbacks +// +// I/O callbacks allow you to read from arbitrary sources, like packaged +// files or some other source. Data read from callbacks are processed +// through a small internal buffer (currently 128 bytes) to try to reduce +// overhead. +// +// The three functions you must define are "read" (reads some bytes of data), +// "skip" (skips some bytes of data), "eof" (reports if the stream is at the end). +// +// =========================================================================== +// +// SIMD support +// +// The JPEG decoder will try to automatically use SIMD kernels on x86 when +// supported by the compiler. For ARM Neon support, you must explicitly +// request it. +// +// (The old do-it-yourself SIMD API is no longer supported in the current +// code.) +// +// On x86, SSE2 will automatically be used when available based on a run-time +// test; if not, the generic C versions are used as a fall-back. On ARM targets, +// the typical path is to have separate builds for NEON and non-NEON devices +// (at least this is true for iOS and Android). Therefore, the NEON support is +// toggled by a build flag: define STBI_NEON to get NEON loops. +// +// If for some reason you do not want to use any of SIMD code, or if +// you have issues compiling it, you can disable it entirely by +// defining STBI_NO_SIMD. +// +// =========================================================================== +// +// HDR image support (disable by defining STBI_NO_HDR) +// +// stb_image supports loading HDR images in general, and currently the Radiance +// .HDR file format specifically. You can still load any file through the existing +// interface; if you attempt to load an HDR file, it will be automatically remapped +// to LDR, assuming gamma 2.2 and an arbitrary scale factor defaulting to 1; +// both of these constants can be reconfigured through this interface: +// +// stbi_hdr_to_ldr_gamma(2.2f); +// stbi_hdr_to_ldr_scale(1.0f); +// +// (note, do not use _inverse_ constants; stbi_image will invert them +// appropriately). +// +// Additionally, there is a new, parallel interface for loading files as +// (linear) floats to preserve the full dynamic range: +// +// float *data = stbi_loadf(filename, &x, &y, &n, 0); +// +// If you load LDR images through this interface, those images will +// be promoted to floating point values, run through the inverse of +// constants corresponding to the above: +// +// stbi_ldr_to_hdr_scale(1.0f); +// stbi_ldr_to_hdr_gamma(2.2f); +// +// Finally, given a filename (or an open file or memory block--see header +// file for details) containing image data, you can query for the "most +// appropriate" interface to use (that is, whether the image is HDR or +// not), using: +// +// stbi_is_hdr(char *filename); +// +// =========================================================================== +// +// iPhone PNG support: +// +// We optionally support converting iPhone-formatted PNGs (which store +// premultiplied BGRA) back to RGB, even though they're internally encoded +// differently. To enable this conversion, call +// stbi_convert_iphone_png_to_rgb(1). +// +// Call stbi_set_unpremultiply_on_load(1) as well to force a divide per +// pixel to remove any premultiplied alpha *only* if the image file explicitly +// says there's premultiplied data (currently only happens in iPhone images, +// and only if iPhone convert-to-rgb processing is on). +// +// =========================================================================== +// +// ADDITIONAL CONFIGURATION +// +// - You can suppress implementation of any of the decoders to reduce +// your code footprint by #defining one or more of the following +// symbols before creating the implementation. +// +// STBI_NO_JPEG +// STBI_NO_PNG +// STBI_NO_BMP +// STBI_NO_PSD +// STBI_NO_TGA +// STBI_NO_GIF +// STBI_NO_HDR +// STBI_NO_PIC +// STBI_NO_PNM (.ppm and .pgm) +// +// - You can request *only* certain decoders and suppress all other ones +// (this will be more forward-compatible, as addition of new decoders +// doesn't require you to disable them explicitly): +// +// STBI_ONLY_JPEG +// STBI_ONLY_PNG +// STBI_ONLY_BMP +// STBI_ONLY_PSD +// STBI_ONLY_TGA +// STBI_ONLY_GIF +// STBI_ONLY_HDR +// STBI_ONLY_PIC +// STBI_ONLY_PNM (.ppm and .pgm) +// +// - If you use STBI_NO_PNG (or _ONLY_ without PNG), and you still +// want the zlib decoder to be available, #define STBI_SUPPORT_ZLIB +// +// - If you define STBI_MAX_DIMENSIONS, stb_image will reject images greater +// than that size (in either width or height) without further processing. +// This is to let programs in the wild set an upper bound to prevent +// denial-of-service attacks on untrusted data, as one could generate a +// valid image of gigantic dimensions and force stb_image to allocate a +// huge block of memory and spend disproportionate time decoding it. By +// default this is set to (1 << 24), which is 16777216, but that's still +// very big. + +# ifndef STBI_NO_STDIO +# include +# endif // STBI_NO_STDIO + +# define STBI_VERSION 1 + +enum { + STBI_default = 0, // only used for desired_channels + + STBI_grey = 1, + STBI_grey_alpha = 2, + STBI_rgb = 3, + STBI_rgb_alpha = 4 +}; + +# include +typedef unsigned char stbi_uc; +typedef unsigned short stbi_us; + +# ifdef __cplusplus +extern "C" { +# endif + +# ifndef STBIDEF +# ifdef STB_IMAGE_STATIC +# define STBIDEF static +# else +# define STBIDEF extern +# endif +# endif + +////////////////////////////////////////////////////////////////////////////// +// +// PRIMARY API - works on images of any type +// + +// +// load image by filename, open file, or memory buffer +// + +typedef struct +{ + int (*read)(void* user, char* data, int size); // fill 'data' with 'size' bytes. return number of bytes actually read + void (*skip)(void* user, int n); // skip the next 'n' bytes, or 'unget' the last -n bytes if negative + int (*eof)(void* user); // returns nonzero if we are at end of file/data +} stbi_io_callbacks; + +//////////////////////////////////// +// +// 8-bits-per-channel interface +// + +STBIDEF stbi_uc* stbi_load_from_memory(stbi_uc const* buffer, int len, int* x, int* y, int* channels_in_file, int desired_channels); +STBIDEF stbi_uc* stbi_load_from_callbacks(stbi_io_callbacks const* clbk, void* user, int* x, int* y, int* channels_in_file, int desired_channels); + +# ifndef STBI_NO_STDIO +STBIDEF stbi_uc* stbi_load(char const* filename, int* x, int* y, int* channels_in_file, int desired_channels); +STBIDEF stbi_uc* stbi_load_from_file(FILE* f, int* x, int* y, int* channels_in_file, int desired_channels); +// for stbi_load_from_file, file pointer is left pointing immediately after image +# endif + +# ifndef STBI_NO_GIF +STBIDEF stbi_uc* stbi_load_gif_from_memory(stbi_uc const* buffer, int len, int** delays, int* x, int* y, int* z, int* comp, int req_comp); +# endif + +# ifdef STBI_WINDOWS_UTF8 +STBIDEF int stbi_convert_wchar_to_utf8(char* buffer, size_t bufferlen, wchar_t const* input); +# endif + +//////////////////////////////////// +// +// 16-bits-per-channel interface +// + +STBIDEF stbi_us* stbi_load_16_from_memory(stbi_uc const* buffer, int len, int* x, int* y, int* channels_in_file, int desired_channels); +STBIDEF stbi_us* stbi_load_16_from_callbacks(stbi_io_callbacks const* clbk, void* user, int* x, int* y, int* channels_in_file, int desired_channels); + +# ifndef STBI_NO_STDIO +STBIDEF stbi_us* stbi_load_16(char const* filename, int* x, int* y, int* channels_in_file, int desired_channels); +STBIDEF stbi_us* stbi_load_from_file_16(FILE* f, int* x, int* y, int* channels_in_file, int desired_channels); +# endif + +//////////////////////////////////// +// +// float-per-channel interface +// +# ifndef STBI_NO_LINEAR +STBIDEF float* stbi_loadf_from_memory(stbi_uc const* buffer, int len, int* x, int* y, int* channels_in_file, int desired_channels); +STBIDEF float* stbi_loadf_from_callbacks(stbi_io_callbacks const* clbk, void* user, int* x, int* y, int* channels_in_file, int desired_channels); + +# ifndef STBI_NO_STDIO +STBIDEF float* stbi_loadf(char const* filename, int* x, int* y, int* channels_in_file, int desired_channels); +STBIDEF float* stbi_loadf_from_file(FILE* f, int* x, int* y, int* channels_in_file, int desired_channels); +# endif +# endif + +# ifndef STBI_NO_HDR +STBIDEF void stbi_hdr_to_ldr_gamma(float gamma); +STBIDEF void stbi_hdr_to_ldr_scale(float scale); +# endif // STBI_NO_HDR + +# ifndef STBI_NO_LINEAR +STBIDEF void stbi_ldr_to_hdr_gamma(float gamma); +STBIDEF void stbi_ldr_to_hdr_scale(float scale); +# endif // STBI_NO_LINEAR + +// stbi_is_hdr is always defined, but always returns false if STBI_NO_HDR +STBIDEF int stbi_is_hdr_from_callbacks(stbi_io_callbacks const* clbk, void* user); +STBIDEF int stbi_is_hdr_from_memory(stbi_uc const* buffer, int len); +# ifndef STBI_NO_STDIO +STBIDEF int stbi_is_hdr(char const* filename); +STBIDEF int stbi_is_hdr_from_file(FILE* f); +# endif // STBI_NO_STDIO + +// get a VERY brief reason for failure +// on most compilers (and ALL modern mainstream compilers) this is threadsafe +STBIDEF const char* stbi_failure_reason(void); + +// free the loaded image -- this is just free() +STBIDEF void stbi_image_free(void* retval_from_stbi_load); + +// get image dimensions & components without fully decoding +STBIDEF int stbi_info_from_memory(stbi_uc const* buffer, int len, int* x, int* y, int* comp); +STBIDEF int stbi_info_from_callbacks(stbi_io_callbacks const* clbk, void* user, int* x, int* y, int* comp); +STBIDEF int stbi_is_16_bit_from_memory(stbi_uc const* buffer, int len); +STBIDEF int stbi_is_16_bit_from_callbacks(stbi_io_callbacks const* clbk, void* user); + +# ifndef STBI_NO_STDIO +STBIDEF int stbi_info(char const* filename, int* x, int* y, int* comp); +STBIDEF int stbi_info_from_file(FILE* f, int* x, int* y, int* comp); +STBIDEF int stbi_is_16_bit(char const* filename); +STBIDEF int stbi_is_16_bit_from_file(FILE* f); +# endif + +// for image formats that explicitly notate that they have premultiplied alpha, +// we just return the colors as stored in the file. set this flag to force +// unpremultiplication. results are undefined if the unpremultiply overflow. +STBIDEF void stbi_set_unpremultiply_on_load(int flag_true_if_should_unpremultiply); + +// indicate whether we should process iphone images back to canonical format, +// or just pass them through "as-is" +STBIDEF void stbi_convert_iphone_png_to_rgb(int flag_true_if_should_convert); + +// flip the image vertically, so the first pixel in the output array is the bottom left +STBIDEF void stbi_set_flip_vertically_on_load(int flag_true_if_should_flip); + +// as above, but only applies to images loaded on the thread that calls the function +// this function is only available if your compiler supports thread-local variables; +// calling it will fail to link if your compiler doesn't +STBIDEF void stbi_set_unpremultiply_on_load_thread(int flag_true_if_should_unpremultiply); +STBIDEF void stbi_convert_iphone_png_to_rgb_thread(int flag_true_if_should_convert); +STBIDEF void stbi_set_flip_vertically_on_load_thread(int flag_true_if_should_flip); + +// ZLIB client - used by PNG, available for other purposes + +STBIDEF char* stbi_zlib_decode_malloc_guesssize(char const* buffer, int len, int initial_size, int* outlen); +STBIDEF char* stbi_zlib_decode_malloc_guesssize_headerflag(char const* buffer, int len, int initial_size, int* outlen, int parse_header); +STBIDEF char* stbi_zlib_decode_malloc(char const* buffer, int len, int* outlen); +STBIDEF int stbi_zlib_decode_buffer(char* obuffer, int olen, char const* ibuffer, int ilen); + +STBIDEF char* stbi_zlib_decode_noheader_malloc(char const* buffer, int len, int* outlen); +STBIDEF int stbi_zlib_decode_noheader_buffer(char* obuffer, int olen, char const* ibuffer, int ilen); + +# ifdef __cplusplus +} +# endif + +// +// +//// end header file ///////////////////////////////////////////////////// +#endif // STBI_INCLUDE_STB_IMAGE_H + +#ifdef STB_IMAGE_IMPLEMENTATION + +# if defined(STBI_ONLY_JPEG) || defined(STBI_ONLY_PNG) || defined(STBI_ONLY_BMP) \ + || defined(STBI_ONLY_TGA) || defined(STBI_ONLY_GIF) || defined(STBI_ONLY_PSD) \ + || defined(STBI_ONLY_HDR) || defined(STBI_ONLY_PIC) || defined(STBI_ONLY_PNM) \ + || defined(STBI_ONLY_ZLIB) +# ifndef STBI_ONLY_JPEG +# define STBI_NO_JPEG +# endif +# ifndef STBI_ONLY_PNG +# define STBI_NO_PNG +# endif +# ifndef STBI_ONLY_BMP +# define STBI_NO_BMP +# endif +# ifndef STBI_ONLY_PSD +# define STBI_NO_PSD +# endif +# ifndef STBI_ONLY_TGA +# define STBI_NO_TGA +# endif +# ifndef STBI_ONLY_GIF +# define STBI_NO_GIF +# endif +# ifndef STBI_ONLY_HDR +# define STBI_NO_HDR +# endif +# ifndef STBI_ONLY_PIC +# define STBI_NO_PIC +# endif +# ifndef STBI_ONLY_PNM +# define STBI_NO_PNM +# endif +# endif + +# if defined(STBI_NO_PNG) && !defined(STBI_SUPPORT_ZLIB) && !defined(STBI_NO_ZLIB) +# define STBI_NO_ZLIB +# endif + +# include +# include +# include // ptrdiff_t on osx +# include +# include + +# if !defined(STBI_NO_LINEAR) || !defined(STBI_NO_HDR) +# include // ldexp, pow +# endif + +# ifndef STBI_NO_STDIO +# include +# endif + +# ifndef STBI_ASSERT +# include +# define STBI_ASSERT(x) assert(x) +# endif + +# ifdef __cplusplus +# define STBI_EXTERN extern "C" +# else +# define STBI_EXTERN extern +# endif + +# ifndef _MSC_VER +# ifdef __cplusplus +# define stbi_inline inline +# else +# define stbi_inline +# endif +# else +# define stbi_inline __forceinline +# endif + +# ifndef STBI_NO_THREAD_LOCALS +# if defined(__cplusplus) && __cplusplus >= 201103L +# define STBI_THREAD_LOCAL thread_local +# elif defined(__GNUC__) && __GNUC__ < 5 +# define STBI_THREAD_LOCAL __thread +# elif defined(_MSC_VER) +# define STBI_THREAD_LOCAL __declspec(thread) +# elif defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L && !defined(__STDC_NO_THREADS__) +# define STBI_THREAD_LOCAL _Thread_local +# endif + +# ifndef STBI_THREAD_LOCAL +# if defined(__GNUC__) +# define STBI_THREAD_LOCAL __thread +# endif +# endif +# endif + +# if defined(_MSC_VER) || defined(__SYMBIAN32__) +typedef unsigned short stbi__uint16; +typedef signed short stbi__int16; +typedef unsigned int stbi__uint32; +typedef signed int stbi__int32; +# else +# include +typedef uint16_t stbi__uint16; +typedef int16_t stbi__int16; +typedef uint32_t stbi__uint32; +typedef int32_t stbi__int32; +# endif + +// should produce compiler error if size is wrong +typedef unsigned char validate_uint32[sizeof(stbi__uint32) == 4 ? 1 : -1]; + +# ifdef _MSC_VER +# define STBI_NOTUSED(v) (void)(v) +# else +# define STBI_NOTUSED(v) (void)sizeof(v) +# endif + +# ifdef _MSC_VER +# define STBI_HAS_LROTL +# endif + +# ifdef STBI_HAS_LROTL +# define stbi_lrot(x, y) _lrotl(x, y) +# else +# define stbi_lrot(x, y) (((x) << (y)) | ((x) >> (-(y) & 31))) +# endif + +# if defined(STBI_MALLOC) && defined(STBI_FREE) && (defined(STBI_REALLOC) || defined(STBI_REALLOC_SIZED)) +// ok +# elif !defined(STBI_MALLOC) && !defined(STBI_FREE) && !defined(STBI_REALLOC) && !defined(STBI_REALLOC_SIZED) +// ok +# else +# error "Must define all or none of STBI_MALLOC, STBI_FREE, and STBI_REALLOC (or STBI_REALLOC_SIZED)." +# endif + +# ifndef STBI_MALLOC +# define STBI_MALLOC(sz) malloc(sz) +# define STBI_REALLOC(p, newsz) realloc(p, newsz) +# define STBI_FREE(p) free(p) +# endif + +# ifndef STBI_REALLOC_SIZED +# define STBI_REALLOC_SIZED(p, oldsz, newsz) STBI_REALLOC(p, newsz) +# endif + +// x86/x64 detection +# if defined(__x86_64__) || defined(_M_X64) +# define STBI__X64_TARGET +# elif defined(__i386) || defined(_M_IX86) +# define STBI__X86_TARGET +# endif + +# if defined(__GNUC__) && defined(STBI__X86_TARGET) && !defined(__SSE2__) && !defined(STBI_NO_SIMD) +// gcc doesn't support sse2 intrinsics unless you compile with -msse2, +// which in turn means it gets to use SSE2 everywhere. This is unfortunate, +// but previous attempts to provide the SSE2 functions with runtime +// detection caused numerous issues. The way architecture extensions are +// exposed in GCC/Clang is, sadly, not really suited for one-file libs. +// New behavior: if compiled with -msse2, we use SSE2 without any +// detection; if not, we don't use it at all. +# define STBI_NO_SIMD +# endif + +# if defined(__MINGW32__) && defined(STBI__X86_TARGET) && !defined(STBI_MINGW_ENABLE_SSE2) && !defined(STBI_NO_SIMD) +// Note that __MINGW32__ doesn't actually mean 32-bit, so we have to avoid STBI__X64_TARGET +// +// 32-bit MinGW wants ESP to be 16-byte aligned, but this is not in the +// Windows ABI and VC++ as well as Windows DLLs don't maintain that invariant. +// As a result, enabling SSE2 on 32-bit MinGW is dangerous when not +// simultaneously enabling "-mstackrealign". +// +// See https://github.com/nothings/stb/issues/81 for more information. +// +// So default to no SSE2 on 32-bit MinGW. If you've read this far and added +// -mstackrealign to your build settings, feel free to #define STBI_MINGW_ENABLE_SSE2. +# define STBI_NO_SIMD +# endif + +# if !defined(STBI_NO_SIMD) && (defined(STBI__X86_TARGET) || defined(STBI__X64_TARGET)) +# define STBI_SSE2 +# include + +# ifdef _MSC_VER + +# if _MSC_VER >= 1400 // not VC6 +# include // __cpuid +static int stbi__cpuid3(void) +{ + int info[4]; + __cpuid(info, 1); + return info[3]; +} +# else +static int stbi__cpuid3(void) +{ + int res; + __asm { + mov eax,1 + cpuid + mov res,edx + } + return res; +} +# endif + +# define STBI_SIMD_ALIGN(type, name) __declspec(align(16)) type name + +# if !defined(STBI_NO_JPEG) && defined(STBI_SSE2) +static int stbi__sse2_available(void) +{ + int info3 = stbi__cpuid3(); + return ((info3 >> 26) & 1) != 0; +} +# endif + +# else // assume GCC-style if not VC++ +# define STBI_SIMD_ALIGN(type, name) type name __attribute__((aligned(16))) + +# if !defined(STBI_NO_JPEG) && defined(STBI_SSE2) +static int stbi__sse2_available(void) +{ + // If we're even attempting to compile this on GCC/Clang, that means + // -msse2 is on, which means the compiler is allowed to use SSE2 + // instructions at will, and so are we. + return 1; +} +# endif + +# endif +# endif + +// ARM NEON +# if defined(STBI_NO_SIMD) && defined(STBI_NEON) +# undef STBI_NEON +# endif + +# ifdef STBI_NEON +# include +# ifdef _MSC_VER +# define STBI_SIMD_ALIGN(type, name) __declspec(align(16)) type name +# else +# define STBI_SIMD_ALIGN(type, name) type name __attribute__((aligned(16))) +# endif +# endif + +# ifndef STBI_SIMD_ALIGN +# define STBI_SIMD_ALIGN(type, name) type name +# endif + +# ifndef STBI_MAX_DIMENSIONS +# define STBI_MAX_DIMENSIONS (1 << 24) +# endif + +/////////////////////////////////////////////// +// +// stbi__context struct and start_xxx functions + +// stbi__context structure is our basic context used by all images, so it +// contains all the IO context, plus some basic image information +typedef struct +{ + stbi__uint32 img_x, img_y; + int img_n, img_out_n; + + stbi_io_callbacks io; + void* io_user_data; + + int read_from_callbacks; + int buflen; + stbi_uc buffer_start[128]; + int callback_already_read; + + stbi_uc *img_buffer, *img_buffer_end; + stbi_uc *img_buffer_original, *img_buffer_original_end; +} stbi__context; + +static void stbi__refill_buffer(stbi__context* s); + +// initialize a memory-decode context +static void stbi__start_mem(stbi__context* s, stbi_uc const* buffer, int len) +{ + s->io.read = NULL; + s->read_from_callbacks = 0; + s->callback_already_read = 0; + s->img_buffer = s->img_buffer_original = (stbi_uc*)buffer; + s->img_buffer_end = s->img_buffer_original_end = (stbi_uc*)buffer + len; +} + +// initialize a callback-based context +static void stbi__start_callbacks(stbi__context* s, stbi_io_callbacks* c, void* user) +{ + s->io = *c; + s->io_user_data = user; + s->buflen = sizeof(s->buffer_start); + s->read_from_callbacks = 1; + s->callback_already_read = 0; + s->img_buffer = s->img_buffer_original = s->buffer_start; + stbi__refill_buffer(s); + s->img_buffer_original_end = s->img_buffer_end; +} + +# ifndef STBI_NO_STDIO + +static int stbi__stdio_read(void* user, char* data, int size) +{ + return (int)fread(data, 1, size, (FILE*)user); +} + +static void stbi__stdio_skip(void* user, int n) +{ + int ch; + fseek((FILE*)user, n, SEEK_CUR); + ch = fgetc((FILE*)user); /* have to read a byte to reset feof()'s flag */ + if (ch != EOF) { + ungetc(ch, (FILE*)user); /* push byte back onto stream if valid. */ + } +} + +static int stbi__stdio_eof(void* user) +{ + return feof((FILE*)user) || ferror((FILE*)user); +} + +static stbi_io_callbacks stbi__stdio_callbacks = { + stbi__stdio_read, + stbi__stdio_skip, + stbi__stdio_eof, +}; + +static void stbi__start_file(stbi__context* s, FILE* f) +{ + stbi__start_callbacks(s, &stbi__stdio_callbacks, (void*)f); +} + +// static void stop_file(stbi__context *s) { } + +# endif // !STBI_NO_STDIO + +static void stbi__rewind(stbi__context* s) +{ + // conceptually rewind SHOULD rewind to the beginning of the stream, + // but we just rewind to the beginning of the initial buffer, because + // we only use it after doing 'test', which only ever looks at at most 92 bytes + s->img_buffer = s->img_buffer_original; + s->img_buffer_end = s->img_buffer_original_end; +} + +enum { + STBI_ORDER_RGB, + STBI_ORDER_BGR +}; + +typedef struct +{ + int bits_per_channel; + int num_channels; + int channel_order; +} stbi__result_info; + +# ifndef STBI_NO_JPEG +static int stbi__jpeg_test(stbi__context* s); +static void* stbi__jpeg_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri); +static int stbi__jpeg_info(stbi__context* s, int* x, int* y, int* comp); +# endif + +# ifndef STBI_NO_PNG +static int stbi__png_test(stbi__context* s); +static void* stbi__png_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri); +static int stbi__png_info(stbi__context* s, int* x, int* y, int* comp); +static int stbi__png_is16(stbi__context* s); +# endif + +# ifndef STBI_NO_BMP +static int stbi__bmp_test(stbi__context* s); +static void* stbi__bmp_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri); +static int stbi__bmp_info(stbi__context* s, int* x, int* y, int* comp); +# endif + +# ifndef STBI_NO_TGA +static int stbi__tga_test(stbi__context* s); +static void* stbi__tga_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri); +static int stbi__tga_info(stbi__context* s, int* x, int* y, int* comp); +# endif + +# ifndef STBI_NO_PSD +static int stbi__psd_test(stbi__context* s); +static void* stbi__psd_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri, int bpc); +static int stbi__psd_info(stbi__context* s, int* x, int* y, int* comp); +static int stbi__psd_is16(stbi__context* s); +# endif + +# ifndef STBI_NO_HDR +static int stbi__hdr_test(stbi__context* s); +static float* stbi__hdr_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri); +static int stbi__hdr_info(stbi__context* s, int* x, int* y, int* comp); +# endif + +# ifndef STBI_NO_PIC +static int stbi__pic_test(stbi__context* s); +static void* stbi__pic_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri); +static int stbi__pic_info(stbi__context* s, int* x, int* y, int* comp); +# endif + +# ifndef STBI_NO_GIF +static int stbi__gif_test(stbi__context* s); +static void* stbi__gif_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri); +static void* stbi__load_gif_main(stbi__context* s, int** delays, int* x, int* y, int* z, int* comp, int req_comp); +static int stbi__gif_info(stbi__context* s, int* x, int* y, int* comp); +# endif + +# ifndef STBI_NO_PNM +static int stbi__pnm_test(stbi__context* s); +static void* stbi__pnm_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri); +static int stbi__pnm_info(stbi__context* s, int* x, int* y, int* comp); +static int stbi__pnm_is16(stbi__context* s); +# endif + +static +# ifdef STBI_THREAD_LOCAL + STBI_THREAD_LOCAL +# endif + char const* stbi__g_failure_reason; + +STBIDEF const char* stbi_failure_reason(void) +{ + return stbi__g_failure_reason; +} + +# ifndef STBI_NO_FAILURE_STRINGS +static int stbi__err(char const* str) +{ + stbi__g_failure_reason = str; + return 0; +} +# endif + +static void* stbi__malloc(size_t size) +{ + return STBI_MALLOC(size); +} + +// stb_image uses ints pervasively, including for offset calculations. +// therefore the largest decoded image size we can support with the +// current code, even on 64-bit targets, is INT_MAX. this is not a +// significant limitation for the intended use case. +// +// we do, however, need to make sure our size calculations don't +// overflow. hence a few helper functions for size calculations that +// multiply integers together, making sure that they're non-negative +// and no overflow occurs. + +// return 1 if the sum is valid, 0 on overflow. +// negative terms are considered invalid. +static int stbi__addsizes_valid(int a, int b) +{ + if (b < 0) + return 0; + // now 0 <= b <= INT_MAX, hence also + // 0 <= INT_MAX - b <= INTMAX. + // And "a + b <= INT_MAX" (which might overflow) is the + // same as a <= INT_MAX - b (no overflow) + return a <= INT_MAX - b; +} + +// returns 1 if the product is valid, 0 on overflow. +// negative factors are considered invalid. +static int stbi__mul2sizes_valid(int a, int b) +{ + if (a < 0 || b < 0) + return 0; + if (b == 0) + return 1; // mul-by-0 is always safe + // portable way to check for no overflows in a*b + return a <= INT_MAX / b; +} + +# if !defined(STBI_NO_JPEG) || !defined(STBI_NO_PNG) || !defined(STBI_NO_TGA) || !defined(STBI_NO_HDR) +// returns 1 if "a*b + add" has no negative terms/factors and doesn't overflow +static int stbi__mad2sizes_valid(int a, int b, int add) +{ + return stbi__mul2sizes_valid(a, b) && stbi__addsizes_valid(a * b, add); +} +# endif + +// returns 1 if "a*b*c + add" has no negative terms/factors and doesn't overflow +static int stbi__mad3sizes_valid(int a, int b, int c, int add) +{ + return stbi__mul2sizes_valid(a, b) && stbi__mul2sizes_valid(a * b, c) && stbi__addsizes_valid(a * b * c, add); +} + +// returns 1 if "a*b*c*d + add" has no negative terms/factors and doesn't overflow +# if !defined(STBI_NO_LINEAR) || !defined(STBI_NO_HDR) || !defined(STBI_NO_PNM) +static int stbi__mad4sizes_valid(int a, int b, int c, int d, int add) +{ + return stbi__mul2sizes_valid(a, b) && stbi__mul2sizes_valid(a * b, c) && stbi__mul2sizes_valid(a * b * c, d) && stbi__addsizes_valid(a * b * c * d, add); +} +# endif + +# if !defined(STBI_NO_JPEG) || !defined(STBI_NO_PNG) || !defined(STBI_NO_TGA) || !defined(STBI_NO_HDR) +// mallocs with size overflow checking +static void* stbi__malloc_mad2(int a, int b, int add) +{ + if (!stbi__mad2sizes_valid(a, b, add)) + return NULL; + return stbi__malloc(a * b + add); +} +# endif + +static void* stbi__malloc_mad3(int a, int b, int c, int add) +{ + if (!stbi__mad3sizes_valid(a, b, c, add)) + return NULL; + return stbi__malloc(a * b * c + add); +} + +# if !defined(STBI_NO_LINEAR) || !defined(STBI_NO_HDR) || !defined(STBI_NO_PNM) +static void* stbi__malloc_mad4(int a, int b, int c, int d, int add) +{ + if (!stbi__mad4sizes_valid(a, b, c, d, add)) + return NULL; + return stbi__malloc(a * b * c * d + add); +} +# endif + +// returns 1 if the sum of two signed ints is valid (between -2^31 and 2^31-1 inclusive), 0 on overflow. +static int stbi__addints_valid(int a, int b) +{ + if ((a >= 0) != (b >= 0)) + return 1; // a and b have different signs, so no overflow + if (a < 0 && b < 0) + return a >= INT_MIN - b; // same as a + b >= INT_MIN; INT_MIN - b cannot overflow since b < 0. + return a <= INT_MAX - b; +} + +// returns 1 if the product of two ints fits in a signed short, 0 on overflow. +static int stbi__mul2shorts_valid(int a, int b) +{ + if (b == 0 || b == -1) + return 1; // multiplication by 0 is always 0; check for -1 so SHRT_MIN/b doesn't overflow + if ((a >= 0) == (b >= 0)) + return a <= SHRT_MAX / b; // product is positive, so similar to mul2sizes_valid + if (b < 0) + return a <= SHRT_MIN / b; // same as a * b >= SHRT_MIN + return a >= SHRT_MIN / b; +} + +// stbi__err - error +// stbi__errpf - error returning pointer to float +// stbi__errpuc - error returning pointer to unsigned char + +# ifdef STBI_NO_FAILURE_STRINGS +# define stbi__err(x, y) 0 +# elif defined(STBI_FAILURE_USERMSG) +# define stbi__err(x, y) stbi__err(y) +# else +# define stbi__err(x, y) stbi__err(x) +# endif + +# define stbi__errpf(x, y) ((float*)(size_t)(stbi__err(x, y) ? NULL : NULL)) +# define stbi__errpuc(x, y) ((unsigned char*)(size_t)(stbi__err(x, y) ? NULL : NULL)) + +STBIDEF void stbi_image_free(void* retval_from_stbi_load) +{ + STBI_FREE(retval_from_stbi_load); +} + +# ifndef STBI_NO_LINEAR +static float* stbi__ldr_to_hdr(stbi_uc* data, int x, int y, int comp); +# endif + +# ifndef STBI_NO_HDR +static stbi_uc* stbi__hdr_to_ldr(float* data, int x, int y, int comp); +# endif + +static int stbi__vertically_flip_on_load_global = 0; + +STBIDEF void stbi_set_flip_vertically_on_load(int flag_true_if_should_flip) +{ + stbi__vertically_flip_on_load_global = flag_true_if_should_flip; +} + +# ifndef STBI_THREAD_LOCAL +# define stbi__vertically_flip_on_load stbi__vertically_flip_on_load_global +# else +static STBI_THREAD_LOCAL int stbi__vertically_flip_on_load_local, stbi__vertically_flip_on_load_set; + +STBIDEF void stbi_set_flip_vertically_on_load_thread(int flag_true_if_should_flip) +{ + stbi__vertically_flip_on_load_local = flag_true_if_should_flip; + stbi__vertically_flip_on_load_set = 1; +} + +# define stbi__vertically_flip_on_load (stbi__vertically_flip_on_load_set \ + ? stbi__vertically_flip_on_load_local \ + : stbi__vertically_flip_on_load_global) +# endif // STBI_THREAD_LOCAL + +static void* stbi__load_main(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri, int bpc) +{ + memset(ri, 0, sizeof(*ri)); // make sure it's initialized if we add new fields + ri->bits_per_channel = 8; // default is 8 so most paths don't have to be changed + ri->channel_order = STBI_ORDER_RGB; // all current input & output are this, but this is here so we can add BGR order + ri->num_channels = 0; + +// test the formats with a very explicit header first (at least a FOURCC +// or distinctive magic number first) +# ifndef STBI_NO_PNG + if (stbi__png_test(s)) + return stbi__png_load(s, x, y, comp, req_comp, ri); +# endif +# ifndef STBI_NO_BMP + if (stbi__bmp_test(s)) + return stbi__bmp_load(s, x, y, comp, req_comp, ri); +# endif +# ifndef STBI_NO_GIF + if (stbi__gif_test(s)) + return stbi__gif_load(s, x, y, comp, req_comp, ri); +# endif +# ifndef STBI_NO_PSD + if (stbi__psd_test(s)) + return stbi__psd_load(s, x, y, comp, req_comp, ri, bpc); +# else + STBI_NOTUSED(bpc); +# endif +# ifndef STBI_NO_PIC + if (stbi__pic_test(s)) + return stbi__pic_load(s, x, y, comp, req_comp, ri); +# endif + +// then the formats that can end up attempting to load with just 1 or 2 +// bytes matching expectations; these are prone to false positives, so +// try them later +# ifndef STBI_NO_JPEG + if (stbi__jpeg_test(s)) + return stbi__jpeg_load(s, x, y, comp, req_comp, ri); +# endif +# ifndef STBI_NO_PNM + if (stbi__pnm_test(s)) + return stbi__pnm_load(s, x, y, comp, req_comp, ri); +# endif + +# ifndef STBI_NO_HDR + if (stbi__hdr_test(s)) { + float* hdr = stbi__hdr_load(s, x, y, comp, req_comp, ri); + return stbi__hdr_to_ldr(hdr, *x, *y, req_comp ? req_comp : *comp); + } +# endif + +# ifndef STBI_NO_TGA + // test tga last because it's a crappy test! + if (stbi__tga_test(s)) + return stbi__tga_load(s, x, y, comp, req_comp, ri); +# endif + + return stbi__errpuc("unknown image type", "Image not of any known type, or corrupt"); +} + +static stbi_uc* stbi__convert_16_to_8(stbi__uint16* orig, int w, int h, int channels) +{ + int i; + int img_len = w * h * channels; + stbi_uc* reduced; + + reduced = (stbi_uc*)stbi__malloc(img_len); + if (reduced == NULL) + return stbi__errpuc("outofmem", "Out of memory"); + + for (i = 0; i < img_len; ++i) + reduced[i] = (stbi_uc)((orig[i] >> 8) & 0xFF); // top half of each byte is sufficient approx of 16->8 bit scaling + + STBI_FREE(orig); + return reduced; +} + +static stbi__uint16* stbi__convert_8_to_16(stbi_uc* orig, int w, int h, int channels) +{ + int i; + int img_len = w * h * channels; + stbi__uint16* enlarged; + + enlarged = (stbi__uint16*)stbi__malloc(img_len * 2); + if (enlarged == NULL) + return (stbi__uint16*)stbi__errpuc("outofmem", "Out of memory"); + + for (i = 0; i < img_len; ++i) + enlarged[i] = (stbi__uint16)((orig[i] << 8) + orig[i]); // replicate to high and low byte, maps 0->0, 255->0xffff + + STBI_FREE(orig); + return enlarged; +} + +static void stbi__vertical_flip(void* image, int w, int h, int bytes_per_pixel) +{ + return; + int row; + size_t bytes_per_row = (size_t)w * bytes_per_pixel; + stbi_uc temp[2048]; + stbi_uc* bytes = (stbi_uc*)image; + + for (row = 0; row < (h >> 1); row++) { + stbi_uc* row0 = bytes + row * bytes_per_row; + stbi_uc* row1 = bytes + (h - row - 1) * bytes_per_row; + // swap row0 with row1 + size_t bytes_left = bytes_per_row; + while (bytes_left) { + size_t bytes_copy = (bytes_left < sizeof(temp)) ? bytes_left : sizeof(temp); + memcpy(temp, row0, bytes_copy); + memcpy(row0, row1, bytes_copy); + memcpy(row1, temp, bytes_copy); + row0 += bytes_copy; + row1 += bytes_copy; + bytes_left -= bytes_copy; + } + } +} + +# ifndef STBI_NO_GIF +static void stbi__vertical_flip_slices(void* image, int w, int h, int z, int bytes_per_pixel) +{ + return; + int slice; + int slice_size = w * h * bytes_per_pixel; + + stbi_uc* bytes = (stbi_uc*)image; + for (slice = 0; slice < z; ++slice) { + stbi__vertical_flip(bytes, w, h, bytes_per_pixel); + bytes += slice_size; + } +} +# endif + +static unsigned char* stbi__load_and_postprocess_8bit(stbi__context* s, int* x, int* y, int* comp, int req_comp) +{ + stbi__result_info ri; + void* result = stbi__load_main(s, x, y, comp, req_comp, &ri, 8); + + if (result == NULL) + return NULL; + + // it is the responsibility of the loaders to make sure we get either 8 or 16 bit. + STBI_ASSERT(ri.bits_per_channel == 8 || ri.bits_per_channel == 16); + + if (ri.bits_per_channel != 8) { + result = stbi__convert_16_to_8((stbi__uint16*)result, *x, *y, req_comp == 0 ? *comp : req_comp); + ri.bits_per_channel = 8; + } + + // @TODO: move stbi__convert_format to here + + if (stbi__vertically_flip_on_load) { + int channels = req_comp ? req_comp : *comp; + stbi__vertical_flip(result, *x, *y, channels * sizeof(stbi_uc)); + } + + return (unsigned char*)result; +} + +static stbi__uint16* stbi__load_and_postprocess_16bit(stbi__context* s, int* x, int* y, int* comp, int req_comp) +{ + stbi__result_info ri; + void* result = stbi__load_main(s, x, y, comp, req_comp, &ri, 16); + + if (result == NULL) + return NULL; + + // it is the responsibility of the loaders to make sure we get either 8 or 16 bit. + STBI_ASSERT(ri.bits_per_channel == 8 || ri.bits_per_channel == 16); + + if (ri.bits_per_channel != 16) { + result = stbi__convert_8_to_16((stbi_uc*)result, *x, *y, req_comp == 0 ? *comp : req_comp); + ri.bits_per_channel = 16; + } + + // @TODO: move stbi__convert_format16 to here + // @TODO: special case RGB-to-Y (and RGBA-to-YA) for 8-bit-to-16-bit case to keep more precision + + if (stbi__vertically_flip_on_load) { + int channels = req_comp ? req_comp : *comp; + stbi__vertical_flip(result, *x, *y, channels * sizeof(stbi__uint16)); + } + + return (stbi__uint16*)result; +} + +# if !defined(STBI_NO_HDR) && !defined(STBI_NO_LINEAR) +static void stbi__float_postprocess(float* result, int* x, int* y, int* comp, int req_comp) +{ + if (stbi__vertically_flip_on_load && result != NULL) { + int channels = req_comp ? req_comp : *comp; + stbi__vertical_flip(result, *x, *y, channels * sizeof(float)); + } +} +# endif + +# ifndef STBI_NO_STDIO + +# if defined(_WIN32) && defined(STBI_WINDOWS_UTF8) +STBI_EXTERN __declspec(dllimport) int __stdcall MultiByteToWideChar(unsigned int cp, unsigned long flags, char const* str, int cbmb, wchar_t* widestr, int cchwide); +STBI_EXTERN __declspec(dllimport) int __stdcall WideCharToMultiByte(unsigned int cp, unsigned long flags, wchar_t const* widestr, int cchwide, char* str, int cbmb, char const* defchar, int* used_default); +# endif + +# if defined(_WIN32) && defined(STBI_WINDOWS_UTF8) +STBIDEF int stbi_convert_wchar_to_utf8(char* buffer, size_t bufferlen, wchar_t const* input) +{ + return WideCharToMultiByte(65001 /* UTF8 */, 0, input, -1, buffer, (int)bufferlen, NULL, NULL); +} +# endif + +static FILE* stbi__fopen(char const* filename, char const* mode) +{ + FILE* f; +# if defined(_WIN32) && defined(STBI_WINDOWS_UTF8) + wchar_t wMode[64]; + wchar_t wFilename[1024]; + if (0 == MultiByteToWideChar(65001 /* UTF8 */, 0, filename, -1, wFilename, sizeof(wFilename) / sizeof(*wFilename))) + return 0; + + if (0 == MultiByteToWideChar(65001 /* UTF8 */, 0, mode, -1, wMode, sizeof(wMode) / sizeof(*wMode))) + return 0; + +# if defined(_MSC_VER) && _MSC_VER >= 1400 + if (0 != _wfopen_s(&f, wFilename, wMode)) + f = 0; +# else + f = _wfopen(wFilename, wMode); +# endif + +# elif defined(_MSC_VER) && _MSC_VER >= 1400 + if (0 != fopen_s(&f, filename, mode)) + f = 0; +# else + f = fopen(filename, mode); +# endif + return f; +} + +STBIDEF stbi_uc* stbi_load(char const* filename, int* x, int* y, int* comp, int req_comp) +{ + FILE* f = stbi__fopen(filename, "rb"); + unsigned char* result; + if (!f) + return stbi__errpuc("can't fopen", "Unable to open file"); + result = stbi_load_from_file(f, x, y, comp, req_comp); + fclose(f); + return result; +} + +STBIDEF stbi_uc* stbi_load_from_file(FILE* f, int* x, int* y, int* comp, int req_comp) +{ + unsigned char* result; + stbi__context s; + stbi__start_file(&s, f); + result = stbi__load_and_postprocess_8bit(&s, x, y, comp, req_comp); + if (result) { + // need to 'unget' all the characters in the IO buffer + fseek(f, -(int)(s.img_buffer_end - s.img_buffer), SEEK_CUR); + } + return result; +} + +STBIDEF stbi__uint16* stbi_load_from_file_16(FILE* f, int* x, int* y, int* comp, int req_comp) +{ + stbi__uint16* result; + stbi__context s; + stbi__start_file(&s, f); + result = stbi__load_and_postprocess_16bit(&s, x, y, comp, req_comp); + if (result) { + // need to 'unget' all the characters in the IO buffer + fseek(f, -(int)(s.img_buffer_end - s.img_buffer), SEEK_CUR); + } + return result; +} + +STBIDEF stbi_us* stbi_load_16(char const* filename, int* x, int* y, int* comp, int req_comp) +{ + FILE* f = stbi__fopen(filename, "rb"); + stbi__uint16* result; + if (!f) + return (stbi_us*)stbi__errpuc("can't fopen", "Unable to open file"); + result = stbi_load_from_file_16(f, x, y, comp, req_comp); + fclose(f); + return result; +} + +# endif //! STBI_NO_STDIO + +STBIDEF stbi_us* stbi_load_16_from_memory(stbi_uc const* buffer, int len, int* x, int* y, int* channels_in_file, int desired_channels) +{ + stbi__context s; + stbi__start_mem(&s, buffer, len); + return stbi__load_and_postprocess_16bit(&s, x, y, channels_in_file, desired_channels); +} + +STBIDEF stbi_us* stbi_load_16_from_callbacks(stbi_io_callbacks const* clbk, void* user, int* x, int* y, int* channels_in_file, int desired_channels) +{ + stbi__context s; + stbi__start_callbacks(&s, (stbi_io_callbacks*)clbk, user); + return stbi__load_and_postprocess_16bit(&s, x, y, channels_in_file, desired_channels); +} + +STBIDEF stbi_uc* stbi_load_from_memory(stbi_uc const* buffer, int len, int* x, int* y, int* comp, int req_comp) +{ + stbi__context s; + stbi__start_mem(&s, buffer, len); + return stbi__load_and_postprocess_8bit(&s, x, y, comp, req_comp); +} + +STBIDEF stbi_uc* stbi_load_from_callbacks(stbi_io_callbacks const* clbk, void* user, int* x, int* y, int* comp, int req_comp) +{ + stbi__context s; + stbi__start_callbacks(&s, (stbi_io_callbacks*)clbk, user); + return stbi__load_and_postprocess_8bit(&s, x, y, comp, req_comp); +} + +# ifndef STBI_NO_GIF +STBIDEF stbi_uc* stbi_load_gif_from_memory(stbi_uc const* buffer, int len, int** delays, int* x, int* y, int* z, int* comp, int req_comp) +{ + unsigned char* result; + stbi__context s; + stbi__start_mem(&s, buffer, len); + + result = (unsigned char*)stbi__load_gif_main(&s, delays, x, y, z, comp, req_comp); + if (stbi__vertically_flip_on_load) { + stbi__vertical_flip_slices(result, *x, *y, *z, *comp); + } + + return result; +} +# endif + +# ifndef STBI_NO_LINEAR +static float* stbi__loadf_main(stbi__context* s, int* x, int* y, int* comp, int req_comp) +{ + unsigned char* data; +# ifndef STBI_NO_HDR + if (stbi__hdr_test(s)) { + stbi__result_info ri; + float* hdr_data = stbi__hdr_load(s, x, y, comp, req_comp, &ri); + if (hdr_data) + stbi__float_postprocess(hdr_data, x, y, comp, req_comp); + return hdr_data; + } +# endif + data = stbi__load_and_postprocess_8bit(s, x, y, comp, req_comp); + if (data) + return stbi__ldr_to_hdr(data, *x, *y, req_comp ? req_comp : *comp); + return stbi__errpf("unknown image type", "Image not of any known type, or corrupt"); +} + +STBIDEF float* stbi_loadf_from_memory(stbi_uc const* buffer, int len, int* x, int* y, int* comp, int req_comp) +{ + stbi__context s; + stbi__start_mem(&s, buffer, len); + return stbi__loadf_main(&s, x, y, comp, req_comp); +} + +STBIDEF float* stbi_loadf_from_callbacks(stbi_io_callbacks const* clbk, void* user, int* x, int* y, int* comp, int req_comp) +{ + stbi__context s; + stbi__start_callbacks(&s, (stbi_io_callbacks*)clbk, user); + return stbi__loadf_main(&s, x, y, comp, req_comp); +} + +# ifndef STBI_NO_STDIO +STBIDEF float* stbi_loadf(char const* filename, int* x, int* y, int* comp, int req_comp) +{ + float* result; + FILE* f = stbi__fopen(filename, "rb"); + if (!f) + return stbi__errpf("can't fopen", "Unable to open file"); + result = stbi_loadf_from_file(f, x, y, comp, req_comp); + fclose(f); + return result; +} + +STBIDEF float* stbi_loadf_from_file(FILE* f, int* x, int* y, int* comp, int req_comp) +{ + stbi__context s; + stbi__start_file(&s, f); + return stbi__loadf_main(&s, x, y, comp, req_comp); +} +# endif // !STBI_NO_STDIO + +# endif // !STBI_NO_LINEAR + +// these is-hdr-or-not is defined independent of whether STBI_NO_LINEAR is +// defined, for API simplicity; if STBI_NO_LINEAR is defined, it always +// reports false! + +STBIDEF int stbi_is_hdr_from_memory(stbi_uc const* buffer, int len) +{ +# ifndef STBI_NO_HDR + stbi__context s; + stbi__start_mem(&s, buffer, len); + return stbi__hdr_test(&s); +# else + STBI_NOTUSED(buffer); + STBI_NOTUSED(len); + return 0; +# endif +} + +# ifndef STBI_NO_STDIO +STBIDEF int stbi_is_hdr(char const* filename) +{ + FILE* f = stbi__fopen(filename, "rb"); + int result = 0; + if (f) { + result = stbi_is_hdr_from_file(f); + fclose(f); + } + return result; +} + +STBIDEF int stbi_is_hdr_from_file(FILE* f) +{ +# ifndef STBI_NO_HDR + long pos = ftell(f); + int res; + stbi__context s; + stbi__start_file(&s, f); + res = stbi__hdr_test(&s); + fseek(f, pos, SEEK_SET); + return res; +# else + STBI_NOTUSED(f); + return 0; +# endif +} +# endif // !STBI_NO_STDIO + +STBIDEF int stbi_is_hdr_from_callbacks(stbi_io_callbacks const* clbk, void* user) +{ +# ifndef STBI_NO_HDR + stbi__context s; + stbi__start_callbacks(&s, (stbi_io_callbacks*)clbk, user); + return stbi__hdr_test(&s); +# else + STBI_NOTUSED(clbk); + STBI_NOTUSED(user); + return 0; +# endif +} + +# ifndef STBI_NO_LINEAR +static float stbi__l2h_gamma = 2.2f, stbi__l2h_scale = 1.0f; + +STBIDEF void stbi_ldr_to_hdr_gamma(float gamma) { stbi__l2h_gamma = gamma; } +STBIDEF void stbi_ldr_to_hdr_scale(float scale) { stbi__l2h_scale = scale; } +# endif + +static float stbi__h2l_gamma_i = 1.0f / 2.2f, stbi__h2l_scale_i = 1.0f; + +STBIDEF void stbi_hdr_to_ldr_gamma(float gamma) { stbi__h2l_gamma_i = 1 / gamma; } +STBIDEF void stbi_hdr_to_ldr_scale(float scale) { stbi__h2l_scale_i = 1 / scale; } + +////////////////////////////////////////////////////////////////////////////// +// +// Common code used by all image loaders +// + +enum { + STBI__SCAN_load = 0, + STBI__SCAN_type, + STBI__SCAN_header +}; + +static void stbi__refill_buffer(stbi__context* s) +{ + int n = (s->io.read)(s->io_user_data, (char*)s->buffer_start, s->buflen); + s->callback_already_read += (int)(s->img_buffer - s->img_buffer_original); + if (n == 0) { + // at end of file, treat same as if from memory, but need to handle case + // where s->img_buffer isn't pointing to safe memory, e.g. 0-byte file + s->read_from_callbacks = 0; + s->img_buffer = s->buffer_start; + s->img_buffer_end = s->buffer_start + 1; + *s->img_buffer = 0; + } else { + s->img_buffer = s->buffer_start; + s->img_buffer_end = s->buffer_start + n; + } +} + +stbi_inline static stbi_uc stbi__get8(stbi__context* s) +{ + if (s->img_buffer < s->img_buffer_end) + return *s->img_buffer++; + if (s->read_from_callbacks) { + stbi__refill_buffer(s); + return *s->img_buffer++; + } + return 0; +} + +# if defined(STBI_NO_JPEG) && defined(STBI_NO_HDR) && defined(STBI_NO_PIC) && defined(STBI_NO_PNM) +// nothing +# else +stbi_inline static int stbi__at_eof(stbi__context* s) +{ + if (s->io.read) { + if (!(s->io.eof)(s->io_user_data)) + return 0; + // if feof() is true, check if buffer = end + // special case: we've only got the special 0 character at the end + if (s->read_from_callbacks == 0) + return 1; + } + + return s->img_buffer >= s->img_buffer_end; +} +# endif + +# if defined(STBI_NO_JPEG) && defined(STBI_NO_PNG) && defined(STBI_NO_BMP) && defined(STBI_NO_PSD) && defined(STBI_NO_TGA) && defined(STBI_NO_GIF) && defined(STBI_NO_PIC) +// nothing +# else +static void stbi__skip(stbi__context* s, int n) +{ + if (n == 0) + return; // already there! + if (n < 0) { + s->img_buffer = s->img_buffer_end; + return; + } + if (s->io.read) { + int blen = (int)(s->img_buffer_end - s->img_buffer); + if (blen < n) { + s->img_buffer = s->img_buffer_end; + (s->io.skip)(s->io_user_data, n - blen); + return; + } + } + s->img_buffer += n; +} +# endif + +# if defined(STBI_NO_PNG) && defined(STBI_NO_TGA) && defined(STBI_NO_HDR) && defined(STBI_NO_PNM) +// nothing +# else +static int stbi__getn(stbi__context* s, stbi_uc* buffer, int n) +{ + if (s->io.read) { + int blen = (int)(s->img_buffer_end - s->img_buffer); + if (blen < n) { + int res, count; + + memcpy(buffer, s->img_buffer, blen); + + count = (s->io.read)(s->io_user_data, (char*)buffer + blen, n - blen); + res = (count == (n - blen)); + s->img_buffer = s->img_buffer_end; + return res; + } + } + + if (s->img_buffer + n <= s->img_buffer_end) { + memcpy(buffer, s->img_buffer, n); + s->img_buffer += n; + return 1; + } else + return 0; +} +# endif + +# if defined(STBI_NO_JPEG) && defined(STBI_NO_PNG) && defined(STBI_NO_PSD) && defined(STBI_NO_PIC) +// nothing +# else +static int stbi__get16be(stbi__context* s) +{ + int z = stbi__get8(s); + return (z << 8) + stbi__get8(s); +} +# endif + +# if defined(STBI_NO_PNG) && defined(STBI_NO_PSD) && defined(STBI_NO_PIC) +// nothing +# else +static stbi__uint32 stbi__get32be(stbi__context* s) +{ + stbi__uint32 z = stbi__get16be(s); + return (z << 16) + stbi__get16be(s); +} +# endif + +# if defined(STBI_NO_BMP) && defined(STBI_NO_TGA) && defined(STBI_NO_GIF) +// nothing +# else +static int stbi__get16le(stbi__context* s) +{ + int z = stbi__get8(s); + return z + (stbi__get8(s) << 8); +} +# endif + +# ifndef STBI_NO_BMP +static stbi__uint32 stbi__get32le(stbi__context* s) +{ + stbi__uint32 z = stbi__get16le(s); + z += (stbi__uint32)stbi__get16le(s) << 16; + return z; +} +# endif + +# define STBI__BYTECAST(x) ((stbi_uc)((x) & 255)) // truncate int to byte without warnings + +# if defined(STBI_NO_JPEG) && defined(STBI_NO_PNG) && defined(STBI_NO_BMP) && defined(STBI_NO_PSD) && defined(STBI_NO_TGA) && defined(STBI_NO_GIF) && defined(STBI_NO_PIC) && defined(STBI_NO_PNM) +// nothing +# else +////////////////////////////////////////////////////////////////////////////// +// +// generic converter from built-in img_n to req_comp +// individual types do this automatically as much as possible (e.g. jpeg +// does all cases internally since it needs to colorspace convert anyway, +// and it never has alpha, so very few cases ). png can automatically +// interleave an alpha=255 channel, but falls back to this for other cases +// +// assume data buffer is malloced, so malloc a new one and free that one +// only failure mode is malloc failing + +static stbi_uc stbi__compute_y(int r, int g, int b) +{ + return (stbi_uc)(((r * 77) + (g * 150) + (29 * b)) >> 8); +} +# endif + +# if defined(STBI_NO_PNG) && defined(STBI_NO_BMP) && defined(STBI_NO_PSD) && defined(STBI_NO_TGA) && defined(STBI_NO_GIF) && defined(STBI_NO_PIC) && defined(STBI_NO_PNM) +// nothing +# else +static unsigned char* stbi__convert_format(unsigned char* data, int img_n, int req_comp, unsigned int x, unsigned int y) +{ + int i, j; + unsigned char* good; + + if (req_comp == img_n) + return data; + STBI_ASSERT(req_comp >= 1 && req_comp <= 4); + + good = (unsigned char*)stbi__malloc_mad3(req_comp, x, y, 0); + if (good == NULL) { + STBI_FREE(data); + return stbi__errpuc("outofmem", "Out of memory"); + } + + for (j = 0; j < (int)y; ++j) { + unsigned char* src = data + j * x * img_n; + unsigned char* dest = good + j * x * req_comp; + +# define STBI__COMBO(a, b) ((a) * 8 + (b)) +# define STBI__CASE(a, b) \ + case STBI__COMBO(a, b): \ + for (i = x - 1; i >= 0; --i, src += a, dest += b) + // convert source image with img_n components to one with req_comp components; + // avoid switch per pixel, so use switch per scanline and massive macros + switch (STBI__COMBO(img_n, req_comp)) { + STBI__CASE(1, 2) + { + dest[0] = src[0]; + dest[1] = 255; + } + break; + STBI__CASE(1, 3) { dest[0] = dest[1] = dest[2] = src[0]; } + break; + STBI__CASE(1, 4) + { + dest[0] = dest[1] = dest[2] = src[0]; + dest[3] = 255; + } + break; + STBI__CASE(2, 1) { dest[0] = src[0]; } + break; + STBI__CASE(2, 3) { dest[0] = dest[1] = dest[2] = src[0]; } + break; + STBI__CASE(2, 4) + { + dest[0] = dest[1] = dest[2] = src[0]; + dest[3] = src[1]; + } + break; + STBI__CASE(3, 4) + { + dest[0] = src[0]; + dest[1] = src[1]; + dest[2] = src[2]; + dest[3] = 255; + } + break; + STBI__CASE(3, 1) { dest[0] = stbi__compute_y(src[0], src[1], src[2]); } + break; + STBI__CASE(3, 2) + { + dest[0] = stbi__compute_y(src[0], src[1], src[2]); + dest[1] = 255; + } + break; + STBI__CASE(4, 1) { dest[0] = stbi__compute_y(src[0], src[1], src[2]); } + break; + STBI__CASE(4, 2) + { + dest[0] = stbi__compute_y(src[0], src[1], src[2]); + dest[1] = src[3]; + } + break; + STBI__CASE(4, 3) + { + dest[0] = src[0]; + dest[1] = src[1]; + dest[2] = src[2]; + } + break; + default: + STBI_ASSERT(0); + STBI_FREE(data); + STBI_FREE(good); + return stbi__errpuc("unsupported", "Unsupported format conversion"); + } +# undef STBI__CASE + } + + STBI_FREE(data); + return good; +} +# endif + +# if defined(STBI_NO_PNG) && defined(STBI_NO_PSD) +// nothing +# else +static stbi__uint16 stbi__compute_y_16(int r, int g, int b) +{ + return (stbi__uint16)(((r * 77) + (g * 150) + (29 * b)) >> 8); +} +# endif + +# if defined(STBI_NO_PNG) && defined(STBI_NO_PSD) +// nothing +# else +static stbi__uint16* stbi__convert_format16(stbi__uint16* data, int img_n, int req_comp, unsigned int x, unsigned int y) +{ + int i, j; + stbi__uint16* good; + + if (req_comp == img_n) + return data; + STBI_ASSERT(req_comp >= 1 && req_comp <= 4); + + good = (stbi__uint16*)stbi__malloc(req_comp * x * y * 2); + if (good == NULL) { + STBI_FREE(data); + return (stbi__uint16*)stbi__errpuc("outofmem", "Out of memory"); + } + + for (j = 0; j < (int)y; ++j) { + stbi__uint16* src = data + j * x * img_n; + stbi__uint16* dest = good + j * x * req_comp; + +# define STBI__COMBO(a, b) ((a) * 8 + (b)) +# define STBI__CASE(a, b) \ + case STBI__COMBO(a, b): \ + for (i = x - 1; i >= 0; --i, src += a, dest += b) + // convert source image with img_n components to one with req_comp components; + // avoid switch per pixel, so use switch per scanline and massive macros + switch (STBI__COMBO(img_n, req_comp)) { + STBI__CASE(1, 2) + { + dest[0] = src[0]; + dest[1] = 0xffff; + } + break; + STBI__CASE(1, 3) { dest[0] = dest[1] = dest[2] = src[0]; } + break; + STBI__CASE(1, 4) + { + dest[0] = dest[1] = dest[2] = src[0]; + dest[3] = 0xffff; + } + break; + STBI__CASE(2, 1) { dest[0] = src[0]; } + break; + STBI__CASE(2, 3) { dest[0] = dest[1] = dest[2] = src[0]; } + break; + STBI__CASE(2, 4) + { + dest[0] = dest[1] = dest[2] = src[0]; + dest[3] = src[1]; + } + break; + STBI__CASE(3, 4) + { + dest[0] = src[0]; + dest[1] = src[1]; + dest[2] = src[2]; + dest[3] = 0xffff; + } + break; + STBI__CASE(3, 1) { dest[0] = stbi__compute_y_16(src[0], src[1], src[2]); } + break; + STBI__CASE(3, 2) + { + dest[0] = stbi__compute_y_16(src[0], src[1], src[2]); + dest[1] = 0xffff; + } + break; + STBI__CASE(4, 1) { dest[0] = stbi__compute_y_16(src[0], src[1], src[2]); } + break; + STBI__CASE(4, 2) + { + dest[0] = stbi__compute_y_16(src[0], src[1], src[2]); + dest[1] = src[3]; + } + break; + STBI__CASE(4, 3) + { + dest[0] = src[0]; + dest[1] = src[1]; + dest[2] = src[2]; + } + break; + default: + STBI_ASSERT(0); + STBI_FREE(data); + STBI_FREE(good); + return (stbi__uint16*)stbi__errpuc("unsupported", "Unsupported format conversion"); + } +# undef STBI__CASE + } + + STBI_FREE(data); + return good; +} +# endif + +# ifndef STBI_NO_LINEAR +static float* stbi__ldr_to_hdr(stbi_uc* data, int x, int y, int comp) +{ + int i, k, n; + float* output; + if (!data) + return NULL; + output = (float*)stbi__malloc_mad4(x, y, comp, sizeof(float), 0); + if (output == NULL) { + STBI_FREE(data); + return stbi__errpf("outofmem", "Out of memory"); + } + // compute number of non-alpha components + if (comp & 1) + n = comp; + else + n = comp - 1; + for (i = 0; i < x * y; ++i) { + for (k = 0; k < n; ++k) { + output[i * comp + k] = (float)(pow(data[i * comp + k] / 255.0f, stbi__l2h_gamma) * stbi__l2h_scale); + } + } + if (n < comp) { + for (i = 0; i < x * y; ++i) { + output[i * comp + n] = data[i * comp + n] / 255.0f; + } + } + STBI_FREE(data); + return output; +} +# endif + +# ifndef STBI_NO_HDR +# define stbi__float2int(x) ((int)(x)) +static stbi_uc* stbi__hdr_to_ldr(float* data, int x, int y, int comp) +{ + int i, k, n; + stbi_uc* output; + if (!data) + return NULL; + output = (stbi_uc*)stbi__malloc_mad3(x, y, comp, 0); + if (output == NULL) { + STBI_FREE(data); + return stbi__errpuc("outofmem", "Out of memory"); + } + // compute number of non-alpha components + if (comp & 1) + n = comp; + else + n = comp - 1; + for (i = 0; i < x * y; ++i) { + for (k = 0; k < n; ++k) { + float z = (float)pow(data[i * comp + k] * stbi__h2l_scale_i, stbi__h2l_gamma_i) * 255 + 0.5f; + if (z < 0) + z = 0; + if (z > 255) + z = 255; + output[i * comp + k] = (stbi_uc)stbi__float2int(z); + } + if (k < comp) { + float z = data[i * comp + k] * 255 + 0.5f; + if (z < 0) + z = 0; + if (z > 255) + z = 255; + output[i * comp + k] = (stbi_uc)stbi__float2int(z); + } + } + STBI_FREE(data); + return output; +} +# endif + +////////////////////////////////////////////////////////////////////////////// +// +// "baseline" JPEG/JFIF decoder +// +// simple implementation +// - doesn't support delayed output of y-dimension +// - simple interface (only one output format: 8-bit interleaved RGB) +// - doesn't try to recover corrupt jpegs +// - doesn't allow partial loading, loading multiple at once +// - still fast on x86 (copying globals into locals doesn't help x86) +// - allocates lots of intermediate memory (full size of all components) +// - non-interleaved case requires this anyway +// - allows good upsampling (see next) +// high-quality +// - upsampled channels are bilinearly interpolated, even across blocks +// - quality integer IDCT derived from IJG's 'slow' +// performance +// - fast huffman; reasonable integer IDCT +// - some SIMD kernels for common paths on targets with SSE2/NEON +// - uses a lot of intermediate memory, could cache poorly + +# ifndef STBI_NO_JPEG + +// huffman decoding acceleration +# define FAST_BITS 9 // larger handles more cases; smaller stomps less cache + +typedef struct +{ + stbi_uc fast[1 << FAST_BITS]; + // weirdly, repacking this into AoS is a 10% speed loss, instead of a win + stbi__uint16 code[256]; + stbi_uc values[256]; + stbi_uc size[257]; + unsigned int maxcode[18]; + int delta[17]; // old 'firstsymbol' - old 'firstcode' +} stbi__huffman; + +typedef struct +{ + stbi__context* s; + stbi__huffman huff_dc[4]; + stbi__huffman huff_ac[4]; + stbi__uint16 dequant[4][64]; + stbi__int16 fast_ac[4][1 << FAST_BITS]; + + // sizes for components, interleaved MCUs + int img_h_max, img_v_max; + int img_mcu_x, img_mcu_y; + int img_mcu_w, img_mcu_h; + + // definition of jpeg image component + struct + { + int id; + int h, v; + int tq; + int hd, ha; + int dc_pred; + + int x, y, w2, h2; + stbi_uc* data; + void *raw_data, *raw_coeff; + stbi_uc* linebuf; + short* coeff; // progressive only + int coeff_w, coeff_h; // number of 8x8 coefficient blocks + } img_comp[4]; + + stbi__uint32 code_buffer; // jpeg entropy-coded buffer + int code_bits; // number of valid bits + unsigned char marker; // marker seen while filling entropy buffer + int nomore; // flag if we saw a marker so must stop + + int progressive; + int spec_start; + int spec_end; + int succ_high; + int succ_low; + int eob_run; + int jfif; + int app14_color_transform; // Adobe APP14 tag + int rgb; + + int scan_n, order[4]; + int restart_interval, todo; + + // kernels + void (*idct_block_kernel)(stbi_uc* out, int out_stride, short data[64]); + void (*YCbCr_to_RGB_kernel)(stbi_uc* out, stbi_uc const* y, stbi_uc const* pcb, stbi_uc const* pcr, int count, int step); + stbi_uc* (*resample_row_hv_2_kernel)(stbi_uc* out, stbi_uc* in_near, stbi_uc* in_far, int w, int hs); +} stbi__jpeg; + +static int stbi__build_huffman(stbi__huffman* h, int* count) +{ + int i, j, k = 0; + unsigned int code; + // build size list for each symbol (from JPEG spec) + for (i = 0; i < 16; ++i) { + for (j = 0; j < count[i]; ++j) { + h->size[k++] = (stbi_uc)(i + 1); + if (k >= 257) + return stbi__err("bad size list", "Corrupt JPEG"); + } + } + h->size[k] = 0; + + // compute actual symbols (from jpeg spec) + code = 0; + k = 0; + for (j = 1; j <= 16; ++j) { + // compute delta to add to code to compute symbol id + h->delta[j] = k - code; + if (h->size[k] == j) { + while (h->size[k] == j) + h->code[k++] = (stbi__uint16)(code++); + if (code - 1 >= (1u << j)) + return stbi__err("bad code lengths", "Corrupt JPEG"); + } + // compute largest code + 1 for this size, preshifted as needed later + h->maxcode[j] = code << (16 - j); + code <<= 1; + } + h->maxcode[j] = 0xffffffff; + + // build non-spec acceleration table; 255 is flag for not-accelerated + memset(h->fast, 255, 1 << FAST_BITS); + for (i = 0; i < k; ++i) { + int s = h->size[i]; + if (s <= FAST_BITS) { + int c = h->code[i] << (FAST_BITS - s); + int m = 1 << (FAST_BITS - s); + for (j = 0; j < m; ++j) { + h->fast[c + j] = (stbi_uc)i; + } + } + } + return 1; +} + +// build a table that decodes both magnitude and value of small ACs in +// one go. +static void stbi__build_fast_ac(stbi__int16* fast_ac, stbi__huffman* h) +{ + int i; + for (i = 0; i < (1 << FAST_BITS); ++i) { + stbi_uc fast = h->fast[i]; + fast_ac[i] = 0; + if (fast < 255) { + int rs = h->values[fast]; + int run = (rs >> 4) & 15; + int magbits = rs & 15; + int len = h->size[fast]; + + if (magbits && len + magbits <= FAST_BITS) { + // magnitude code followed by receive_extend code + int k = ((i << len) & ((1 << FAST_BITS) - 1)) >> (FAST_BITS - magbits); + int m = 1 << (magbits - 1); + if (k < m) + k += (~0U << magbits) + 1; + // if the result is small enough, we can fit it in fast_ac table + if (k >= -128 && k <= 127) + fast_ac[i] = (stbi__int16)((k * 256) + (run * 16) + (len + magbits)); + } + } + } +} + +static void stbi__grow_buffer_unsafe(stbi__jpeg* j) +{ + do { + unsigned int b = j->nomore ? 0 : stbi__get8(j->s); + if (b == 0xff) { + int c = stbi__get8(j->s); + while (c == 0xff) + c = stbi__get8(j->s); // consume fill bytes + if (c != 0) { + j->marker = (unsigned char)c; + j->nomore = 1; + return; + } + } + j->code_buffer |= b << (24 - j->code_bits); + j->code_bits += 8; + } while (j->code_bits <= 24); +} + +// (1 << n) - 1 +static stbi__uint32 const stbi__bmask[17] = { 0, 1, 3, 7, 15, 31, 63, 127, 255, 511, 1023, 2047, 4095, 8191, 16383, 32767, 65535 }; + +// decode a jpeg huffman value from the bitstream +stbi_inline static int stbi__jpeg_huff_decode(stbi__jpeg* j, stbi__huffman* h) +{ + unsigned int temp; + int c, k; + + if (j->code_bits < 16) + stbi__grow_buffer_unsafe(j); + + // look at the top FAST_BITS and determine what symbol ID it is, + // if the code is <= FAST_BITS + c = (j->code_buffer >> (32 - FAST_BITS)) & ((1 << FAST_BITS) - 1); + k = h->fast[c]; + if (k < 255) { + int s = h->size[k]; + if (s > j->code_bits) + return -1; + j->code_buffer <<= s; + j->code_bits -= s; + return h->values[k]; + } + + // naive test is to shift the code_buffer down so k bits are + // valid, then test against maxcode. To speed this up, we've + // preshifted maxcode left so that it has (16-k) 0s at the + // end; in other words, regardless of the number of bits, it + // wants to be compared against something shifted to have 16; + // that way we don't need to shift inside the loop. + temp = j->code_buffer >> 16; + for (k = FAST_BITS + 1;; ++k) + if (temp < h->maxcode[k]) + break; + if (k == 17) { + // error! code not found + j->code_bits -= 16; + return -1; + } + + if (k > j->code_bits) + return -1; + + // convert the huffman code to the symbol id + c = ((j->code_buffer >> (32 - k)) & stbi__bmask[k]) + h->delta[k]; + if (c < 0 || c >= 256) // symbol id out of bounds! + return -1; + STBI_ASSERT((((j->code_buffer) >> (32 - h->size[c])) & stbi__bmask[h->size[c]]) == h->code[c]); + + // convert the id to a symbol + j->code_bits -= k; + j->code_buffer <<= k; + return h->values[c]; +} + +// bias[n] = (-1<code_bits < n) + stbi__grow_buffer_unsafe(j); + if (j->code_bits < n) + return 0; // ran out of bits from stream, return 0s intead of continuing + + sgn = j->code_buffer >> 31; // sign bit always in MSB; 0 if MSB clear (positive), 1 if MSB set (negative) + k = stbi_lrot(j->code_buffer, n); + j->code_buffer = k & ~stbi__bmask[n]; + k &= stbi__bmask[n]; + j->code_bits -= n; + return k + (stbi__jbias[n] & (sgn - 1)); +} + +// get some unsigned bits +stbi_inline static int stbi__jpeg_get_bits(stbi__jpeg* j, int n) +{ + unsigned int k; + if (j->code_bits < n) + stbi__grow_buffer_unsafe(j); + if (j->code_bits < n) + return 0; // ran out of bits from stream, return 0s intead of continuing + k = stbi_lrot(j->code_buffer, n); + j->code_buffer = k & ~stbi__bmask[n]; + k &= stbi__bmask[n]; + j->code_bits -= n; + return k; +} + +stbi_inline static int stbi__jpeg_get_bit(stbi__jpeg* j) +{ + unsigned int k; + if (j->code_bits < 1) + stbi__grow_buffer_unsafe(j); + if (j->code_bits < 1) + return 0; // ran out of bits from stream, return 0s intead of continuing + k = j->code_buffer; + j->code_buffer <<= 1; + --j->code_bits; + return k & 0x80000000; +} + +// given a value that's at position X in the zigzag stream, +// where does it appear in the 8x8 matrix coded as row-major? +static stbi_uc const stbi__jpeg_dezigzag[64 + 15] = { + 0, 1, 8, 16, 9, 2, 3, 10, + 17, 24, 32, 25, 18, 11, 4, 5, + 12, 19, 26, 33, 40, 48, 41, 34, + 27, 20, 13, 6, 7, 14, 21, 28, + 35, 42, 49, 56, 57, 50, 43, 36, + 29, 22, 15, 23, 30, 37, 44, 51, + 58, 59, 52, 45, 38, 31, 39, 46, + 53, 60, 61, 54, 47, 55, 62, 63, + // let corrupt input sample past end + 63, 63, 63, 63, 63, 63, 63, 63, + 63, 63, 63, 63, 63, 63, 63 +}; + +// decode one 64-entry block-- +static int stbi__jpeg_decode_block(stbi__jpeg* j, short data[64], stbi__huffman* hdc, stbi__huffman* hac, stbi__int16* fac, int b, stbi__uint16* dequant) +{ + int diff, dc, k; + int t; + + if (j->code_bits < 16) + stbi__grow_buffer_unsafe(j); + t = stbi__jpeg_huff_decode(j, hdc); + if (t < 0 || t > 15) + return stbi__err("bad huffman code", "Corrupt JPEG"); + + // 0 all the ac values now so we can do it 32-bits at a time + memset(data, 0, 64 * sizeof(data[0])); + + diff = t ? stbi__extend_receive(j, t) : 0; + if (!stbi__addints_valid(j->img_comp[b].dc_pred, diff)) + return stbi__err("bad delta", "Corrupt JPEG"); + dc = j->img_comp[b].dc_pred + diff; + j->img_comp[b].dc_pred = dc; + if (!stbi__mul2shorts_valid(dc, dequant[0])) + return stbi__err("can't merge dc and ac", "Corrupt JPEG"); + data[0] = (short)(dc * dequant[0]); + + // decode AC components, see JPEG spec + k = 1; + do { + unsigned int zig; + int c, r, s; + if (j->code_bits < 16) + stbi__grow_buffer_unsafe(j); + c = (j->code_buffer >> (32 - FAST_BITS)) & ((1 << FAST_BITS) - 1); + r = fac[c]; + if (r) { // fast-AC path + k += (r >> 4) & 15; // run + s = r & 15; // combined length + if (s > j->code_bits) + return stbi__err("bad huffman code", "Combined length longer than code bits available"); + j->code_buffer <<= s; + j->code_bits -= s; + // decode into unzigzag'd location + zig = stbi__jpeg_dezigzag[k++]; + data[zig] = (short)((r >> 8) * dequant[zig]); + } else { + int rs = stbi__jpeg_huff_decode(j, hac); + if (rs < 0) + return stbi__err("bad huffman code", "Corrupt JPEG"); + s = rs & 15; + r = rs >> 4; + if (s == 0) { + if (rs != 0xf0) + break; // end block + k += 16; + } else { + k += r; + // decode into unzigzag'd location + zig = stbi__jpeg_dezigzag[k++]; + data[zig] = (short)(stbi__extend_receive(j, s) * dequant[zig]); + } + } + } while (k < 64); + return 1; +} + +static int stbi__jpeg_decode_block_prog_dc(stbi__jpeg* j, short data[64], stbi__huffman* hdc, int b) +{ + int diff, dc; + int t; + if (j->spec_end != 0) + return stbi__err("can't merge dc and ac", "Corrupt JPEG"); + + if (j->code_bits < 16) + stbi__grow_buffer_unsafe(j); + + if (j->succ_high == 0) { + // first scan for DC coefficient, must be first + memset(data, 0, 64 * sizeof(data[0])); // 0 all the ac values now + t = stbi__jpeg_huff_decode(j, hdc); + if (t < 0 || t > 15) + return stbi__err("can't merge dc and ac", "Corrupt JPEG"); + diff = t ? stbi__extend_receive(j, t) : 0; + + if (!stbi__addints_valid(j->img_comp[b].dc_pred, diff)) + return stbi__err("bad delta", "Corrupt JPEG"); + dc = j->img_comp[b].dc_pred + diff; + j->img_comp[b].dc_pred = dc; + if (!stbi__mul2shorts_valid(dc, 1 << j->succ_low)) + return stbi__err("can't merge dc and ac", "Corrupt JPEG"); + data[0] = (short)(dc * (1 << j->succ_low)); + } else { + // refinement scan for DC coefficient + if (stbi__jpeg_get_bit(j)) + data[0] += (short)(1 << j->succ_low); + } + return 1; +} + +// @OPTIMIZE: store non-zigzagged during the decode passes, +// and only de-zigzag when dequantizing +static int stbi__jpeg_decode_block_prog_ac(stbi__jpeg* j, short data[64], stbi__huffman* hac, stbi__int16* fac) +{ + int k; + if (j->spec_start == 0) + return stbi__err("can't merge dc and ac", "Corrupt JPEG"); + + if (j->succ_high == 0) { + int shift = j->succ_low; + + if (j->eob_run) { + --j->eob_run; + return 1; + } + + k = j->spec_start; + do { + unsigned int zig; + int c, r, s; + if (j->code_bits < 16) + stbi__grow_buffer_unsafe(j); + c = (j->code_buffer >> (32 - FAST_BITS)) & ((1 << FAST_BITS) - 1); + r = fac[c]; + if (r) { // fast-AC path + k += (r >> 4) & 15; // run + s = r & 15; // combined length + if (s > j->code_bits) + return stbi__err("bad huffman code", "Combined length longer than code bits available"); + j->code_buffer <<= s; + j->code_bits -= s; + zig = stbi__jpeg_dezigzag[k++]; + data[zig] = (short)((r >> 8) * (1 << shift)); + } else { + int rs = stbi__jpeg_huff_decode(j, hac); + if (rs < 0) + return stbi__err("bad huffman code", "Corrupt JPEG"); + s = rs & 15; + r = rs >> 4; + if (s == 0) { + if (r < 15) { + j->eob_run = (1 << r); + if (r) + j->eob_run += stbi__jpeg_get_bits(j, r); + --j->eob_run; + break; + } + k += 16; + } else { + k += r; + zig = stbi__jpeg_dezigzag[k++]; + data[zig] = (short)(stbi__extend_receive(j, s) * (1 << shift)); + } + } + } while (k <= j->spec_end); + } else { + // refinement scan for these AC coefficients + + short bit = (short)(1 << j->succ_low); + + if (j->eob_run) { + --j->eob_run; + for (k = j->spec_start; k <= j->spec_end; ++k) { + short* p = &data[stbi__jpeg_dezigzag[k]]; + if (*p != 0) + if (stbi__jpeg_get_bit(j)) + if ((*p & bit) == 0) { + if (*p > 0) + *p += bit; + else + *p -= bit; + } + } + } else { + k = j->spec_start; + do { + int r, s; + int rs = stbi__jpeg_huff_decode(j, hac); // @OPTIMIZE see if we can use the fast path here, advance-by-r is so slow, eh + if (rs < 0) + return stbi__err("bad huffman code", "Corrupt JPEG"); + s = rs & 15; + r = rs >> 4; + if (s == 0) { + if (r < 15) { + j->eob_run = (1 << r) - 1; + if (r) + j->eob_run += stbi__jpeg_get_bits(j, r); + r = 64; // force end of block + } else { + // r=15 s=0 should write 16 0s, so we just do + // a run of 15 0s and then write s (which is 0), + // so we don't have to do anything special here + } + } else { + if (s != 1) + return stbi__err("bad huffman code", "Corrupt JPEG"); + // sign bit + if (stbi__jpeg_get_bit(j)) + s = bit; + else + s = -bit; + } + + // advance by r + while (k <= j->spec_end) { + short* p = &data[stbi__jpeg_dezigzag[k++]]; + if (*p != 0) { + if (stbi__jpeg_get_bit(j)) + if ((*p & bit) == 0) { + if (*p > 0) + *p += bit; + else + *p -= bit; + } + } else { + if (r == 0) { + *p = (short)s; + break; + } + --r; + } + } + } while (k <= j->spec_end); + } + } + return 1; +} + +// take a -128..127 value and stbi__clamp it and convert to 0..255 +stbi_inline static stbi_uc stbi__clamp(int x) +{ + // trick to use a single test to catch both cases + if ((unsigned int)x > 255) { + if (x < 0) + return 0; + if (x > 255) + return 255; + } + return (stbi_uc)x; +} + +# define stbi__f2f(x) ((int)(((x) * 4096 + 0.5))) +# define stbi__fsh(x) ((x) * 4096) + +// derived from jidctint -- DCT_ISLOW +# define STBI__IDCT_1D(s0, s1, s2, s3, s4, s5, s6, s7) \ + int t0, t1, t2, t3, p1, p2, p3, p4, p5, x0, x1, x2, x3; \ + p2 = s2; \ + p3 = s6; \ + p1 = (p2 + p3) * stbi__f2f(0.5411961f); \ + t2 = p1 + p3 * stbi__f2f(-1.847759065f); \ + t3 = p1 + p2 * stbi__f2f(0.765366865f); \ + p2 = s0; \ + p3 = s4; \ + t0 = stbi__fsh(p2 + p3); \ + t1 = stbi__fsh(p2 - p3); \ + x0 = t0 + t3; \ + x3 = t0 - t3; \ + x1 = t1 + t2; \ + x2 = t1 - t2; \ + t0 = s7; \ + t1 = s5; \ + t2 = s3; \ + t3 = s1; \ + p3 = t0 + t2; \ + p4 = t1 + t3; \ + p1 = t0 + t3; \ + p2 = t1 + t2; \ + p5 = (p3 + p4) * stbi__f2f(1.175875602f); \ + t0 = t0 * stbi__f2f(0.298631336f); \ + t1 = t1 * stbi__f2f(2.053119869f); \ + t2 = t2 * stbi__f2f(3.072711026f); \ + t3 = t3 * stbi__f2f(1.501321110f); \ + p1 = p5 + p1 * stbi__f2f(-0.899976223f); \ + p2 = p5 + p2 * stbi__f2f(-2.562915447f); \ + p3 = p3 * stbi__f2f(-1.961570560f); \ + p4 = p4 * stbi__f2f(-0.390180644f); \ + t3 += p1 + p4; \ + t2 += p2 + p3; \ + t1 += p2 + p4; \ + t0 += p1 + p3; + +static void stbi__idct_block(stbi_uc* out, int out_stride, short data[64]) +{ + int i, val[64], *v = val; + stbi_uc* o; + short* d = data; + + // columns + for (i = 0; i < 8; ++i, ++d, ++v) { + // if all zeroes, shortcut -- this avoids dequantizing 0s and IDCTing + if (d[8] == 0 && d[16] == 0 && d[24] == 0 && d[32] == 0 + && d[40] == 0 && d[48] == 0 && d[56] == 0) { + // no shortcut 0 seconds + // (1|2|3|4|5|6|7)==0 0 seconds + // all separate -0.047 seconds + // 1 && 2|3 && 4|5 && 6|7: -0.047 seconds + int dcterm = d[0] * 4; + v[0] = v[8] = v[16] = v[24] = v[32] = v[40] = v[48] = v[56] = dcterm; + } else { + STBI__IDCT_1D(d[0], d[8], d[16], d[24], d[32], d[40], d[48], d[56]) + // constants scaled things up by 1<<12; let's bring them back + // down, but keep 2 extra bits of precision + x0 += 512; + x1 += 512; + x2 += 512; + x3 += 512; + v[0] = (x0 + t3) >> 10; + v[56] = (x0 - t3) >> 10; + v[8] = (x1 + t2) >> 10; + v[48] = (x1 - t2) >> 10; + v[16] = (x2 + t1) >> 10; + v[40] = (x2 - t1) >> 10; + v[24] = (x3 + t0) >> 10; + v[32] = (x3 - t0) >> 10; + } + } + + for (i = 0, v = val, o = out; i < 8; ++i, v += 8, o += out_stride) { + // no fast case since the first 1D IDCT spread components out + STBI__IDCT_1D(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7]) + // constants scaled things up by 1<<12, plus we had 1<<2 from first + // loop, plus horizontal and vertical each scale by sqrt(8) so together + // we've got an extra 1<<3, so 1<<17 total we need to remove. + // so we want to round that, which means adding 0.5 * 1<<17, + // aka 65536. Also, we'll end up with -128 to 127 that we want + // to encode as 0..255 by adding 128, so we'll add that before the shift + x0 += 65536 + (128 << 17); + x1 += 65536 + (128 << 17); + x2 += 65536 + (128 << 17); + x3 += 65536 + (128 << 17); + // tried computing the shifts into temps, or'ing the temps to see + // if any were out of range, but that was slower + o[0] = stbi__clamp((x0 + t3) >> 17); + o[7] = stbi__clamp((x0 - t3) >> 17); + o[1] = stbi__clamp((x1 + t2) >> 17); + o[6] = stbi__clamp((x1 - t2) >> 17); + o[2] = stbi__clamp((x2 + t1) >> 17); + o[5] = stbi__clamp((x2 - t1) >> 17); + o[3] = stbi__clamp((x3 + t0) >> 17); + o[4] = stbi__clamp((x3 - t0) >> 17); + } +} + +# ifdef STBI_SSE2 +// sse2 integer IDCT. not the fastest possible implementation but it +// produces bit-identical results to the generic C version so it's +// fully "transparent". +static void stbi__idct_simd(stbi_uc* out, int out_stride, short data[64]) +{ + // This is constructed to match our regular (generic) integer IDCT exactly. + __m128i row0, row1, row2, row3, row4, row5, row6, row7; + __m128i tmp; + +// dot product constant: even elems=x, odd elems=y +# define dct_const(x, y) _mm_setr_epi16((x), (y), (x), (y), (x), (y), (x), (y)) + +// out(0) = c0[even]*x + c0[odd]*y (c0, x, y 16-bit, out 32-bit) +// out(1) = c1[even]*x + c1[odd]*y +# define dct_rot(out0, out1, x, y, c0, c1) \ + __m128i c0##lo = _mm_unpacklo_epi16((x), (y)); \ + __m128i c0##hi = _mm_unpackhi_epi16((x), (y)); \ + __m128i out0##_l = _mm_madd_epi16(c0##lo, c0); \ + __m128i out0##_h = _mm_madd_epi16(c0##hi, c0); \ + __m128i out1##_l = _mm_madd_epi16(c0##lo, c1); \ + __m128i out1##_h = _mm_madd_epi16(c0##hi, c1) + +// out = in << 12 (in 16-bit, out 32-bit) +# define dct_widen(out, in) \ + __m128i out##_l = _mm_srai_epi32(_mm_unpacklo_epi16(_mm_setzero_si128(), (in)), 4); \ + __m128i out##_h = _mm_srai_epi32(_mm_unpackhi_epi16(_mm_setzero_si128(), (in)), 4) + +// wide add +# define dct_wadd(out, a, b) \ + __m128i out##_l = _mm_add_epi32(a##_l, b##_l); \ + __m128i out##_h = _mm_add_epi32(a##_h, b##_h) + +// wide sub +# define dct_wsub(out, a, b) \ + __m128i out##_l = _mm_sub_epi32(a##_l, b##_l); \ + __m128i out##_h = _mm_sub_epi32(a##_h, b##_h) + +// butterfly a/b, add bias, then shift by "s" and pack +# define dct_bfly32o(out0, out1, a, b, bias, s) \ + { \ + __m128i abiased_l = _mm_add_epi32(a##_l, bias); \ + __m128i abiased_h = _mm_add_epi32(a##_h, bias); \ + dct_wadd(sum, abiased, b); \ + dct_wsub(dif, abiased, b); \ + out0 = _mm_packs_epi32(_mm_srai_epi32(sum_l, s), _mm_srai_epi32(sum_h, s)); \ + out1 = _mm_packs_epi32(_mm_srai_epi32(dif_l, s), _mm_srai_epi32(dif_h, s)); \ + } + +// 8-bit interleave step (for transposes) +# define dct_interleave8(a, b) \ + tmp = a; \ + a = _mm_unpacklo_epi8(a, b); \ + b = _mm_unpackhi_epi8(tmp, b) + +// 16-bit interleave step (for transposes) +# define dct_interleave16(a, b) \ + tmp = a; \ + a = _mm_unpacklo_epi16(a, b); \ + b = _mm_unpackhi_epi16(tmp, b) + +# define dct_pass(bias, shift) \ + { \ + /* even part */ \ + dct_rot(t2e, t3e, row2, row6, rot0_0, rot0_1); \ + __m128i sum04 = _mm_add_epi16(row0, row4); \ + __m128i dif04 = _mm_sub_epi16(row0, row4); \ + dct_widen(t0e, sum04); \ + dct_widen(t1e, dif04); \ + dct_wadd(x0, t0e, t3e); \ + dct_wsub(x3, t0e, t3e); \ + dct_wadd(x1, t1e, t2e); \ + dct_wsub(x2, t1e, t2e); \ + /* odd part */ \ + dct_rot(y0o, y2o, row7, row3, rot2_0, rot2_1); \ + dct_rot(y1o, y3o, row5, row1, rot3_0, rot3_1); \ + __m128i sum17 = _mm_add_epi16(row1, row7); \ + __m128i sum35 = _mm_add_epi16(row3, row5); \ + dct_rot(y4o, y5o, sum17, sum35, rot1_0, rot1_1); \ + dct_wadd(x4, y0o, y4o); \ + dct_wadd(x5, y1o, y5o); \ + dct_wadd(x6, y2o, y5o); \ + dct_wadd(x7, y3o, y4o); \ + dct_bfly32o(row0, row7, x0, x7, bias, shift); \ + dct_bfly32o(row1, row6, x1, x6, bias, shift); \ + dct_bfly32o(row2, row5, x2, x5, bias, shift); \ + dct_bfly32o(row3, row4, x3, x4, bias, shift); \ + } + + __m128i rot0_0 = dct_const(stbi__f2f(0.5411961f), stbi__f2f(0.5411961f) + stbi__f2f(-1.847759065f)); + __m128i rot0_1 = dct_const(stbi__f2f(0.5411961f) + stbi__f2f(0.765366865f), stbi__f2f(0.5411961f)); + __m128i rot1_0 = dct_const(stbi__f2f(1.175875602f) + stbi__f2f(-0.899976223f), stbi__f2f(1.175875602f)); + __m128i rot1_1 = dct_const(stbi__f2f(1.175875602f), stbi__f2f(1.175875602f) + stbi__f2f(-2.562915447f)); + __m128i rot2_0 = dct_const(stbi__f2f(-1.961570560f) + stbi__f2f(0.298631336f), stbi__f2f(-1.961570560f)); + __m128i rot2_1 = dct_const(stbi__f2f(-1.961570560f), stbi__f2f(-1.961570560f) + stbi__f2f(3.072711026f)); + __m128i rot3_0 = dct_const(stbi__f2f(-0.390180644f) + stbi__f2f(2.053119869f), stbi__f2f(-0.390180644f)); + __m128i rot3_1 = dct_const(stbi__f2f(-0.390180644f), stbi__f2f(-0.390180644f) + stbi__f2f(1.501321110f)); + + // rounding biases in column/row passes, see stbi__idct_block for explanation. + __m128i bias_0 = _mm_set1_epi32(512); + __m128i bias_1 = _mm_set1_epi32(65536 + (128 << 17)); + + // load + row0 = _mm_load_si128((__m128i const*)(data + 0 * 8)); + row1 = _mm_load_si128((__m128i const*)(data + 1 * 8)); + row2 = _mm_load_si128((__m128i const*)(data + 2 * 8)); + row3 = _mm_load_si128((__m128i const*)(data + 3 * 8)); + row4 = _mm_load_si128((__m128i const*)(data + 4 * 8)); + row5 = _mm_load_si128((__m128i const*)(data + 5 * 8)); + row6 = _mm_load_si128((__m128i const*)(data + 6 * 8)); + row7 = _mm_load_si128((__m128i const*)(data + 7 * 8)); + + // column pass + dct_pass(bias_0, 10); + + { + // 16bit 8x8 transpose pass 1 + dct_interleave16(row0, row4); + dct_interleave16(row1, row5); + dct_interleave16(row2, row6); + dct_interleave16(row3, row7); + + // transpose pass 2 + dct_interleave16(row0, row2); + dct_interleave16(row1, row3); + dct_interleave16(row4, row6); + dct_interleave16(row5, row7); + + // transpose pass 3 + dct_interleave16(row0, row1); + dct_interleave16(row2, row3); + dct_interleave16(row4, row5); + dct_interleave16(row6, row7); + } + + // row pass + dct_pass(bias_1, 17); + + { + // pack + __m128i p0 = _mm_packus_epi16(row0, row1); // a0a1a2a3...a7b0b1b2b3...b7 + __m128i p1 = _mm_packus_epi16(row2, row3); + __m128i p2 = _mm_packus_epi16(row4, row5); + __m128i p3 = _mm_packus_epi16(row6, row7); + + // 8bit 8x8 transpose pass 1 + dct_interleave8(p0, p2); // a0e0a1e1... + dct_interleave8(p1, p3); // c0g0c1g1... + + // transpose pass 2 + dct_interleave8(p0, p1); // a0c0e0g0... + dct_interleave8(p2, p3); // b0d0f0h0... + + // transpose pass 3 + dct_interleave8(p0, p2); // a0b0c0d0... + dct_interleave8(p1, p3); // a4b4c4d4... + + // store + _mm_storel_epi64((__m128i*)out, p0); + out += out_stride; + _mm_storel_epi64((__m128i*)out, _mm_shuffle_epi32(p0, 0x4e)); + out += out_stride; + _mm_storel_epi64((__m128i*)out, p2); + out += out_stride; + _mm_storel_epi64((__m128i*)out, _mm_shuffle_epi32(p2, 0x4e)); + out += out_stride; + _mm_storel_epi64((__m128i*)out, p1); + out += out_stride; + _mm_storel_epi64((__m128i*)out, _mm_shuffle_epi32(p1, 0x4e)); + out += out_stride; + _mm_storel_epi64((__m128i*)out, p3); + out += out_stride; + _mm_storel_epi64((__m128i*)out, _mm_shuffle_epi32(p3, 0x4e)); + } + +# undef dct_const +# undef dct_rot +# undef dct_widen +# undef dct_wadd +# undef dct_wsub +# undef dct_bfly32o +# undef dct_interleave8 +# undef dct_interleave16 +# undef dct_pass +} + +# endif // STBI_SSE2 + +# ifdef STBI_NEON + +// NEON integer IDCT. should produce bit-identical +// results to the generic C version. +static void stbi__idct_simd(stbi_uc* out, int out_stride, short data[64]) +{ + int16x8_t row0, row1, row2, row3, row4, row5, row6, row7; + + int16x4_t rot0_0 = vdup_n_s16(stbi__f2f(0.5411961f)); + int16x4_t rot0_1 = vdup_n_s16(stbi__f2f(-1.847759065f)); + int16x4_t rot0_2 = vdup_n_s16(stbi__f2f(0.765366865f)); + int16x4_t rot1_0 = vdup_n_s16(stbi__f2f(1.175875602f)); + int16x4_t rot1_1 = vdup_n_s16(stbi__f2f(-0.899976223f)); + int16x4_t rot1_2 = vdup_n_s16(stbi__f2f(-2.562915447f)); + int16x4_t rot2_0 = vdup_n_s16(stbi__f2f(-1.961570560f)); + int16x4_t rot2_1 = vdup_n_s16(stbi__f2f(-0.390180644f)); + int16x4_t rot3_0 = vdup_n_s16(stbi__f2f(0.298631336f)); + int16x4_t rot3_1 = vdup_n_s16(stbi__f2f(2.053119869f)); + int16x4_t rot3_2 = vdup_n_s16(stbi__f2f(3.072711026f)); + int16x4_t rot3_3 = vdup_n_s16(stbi__f2f(1.501321110f)); + +# define dct_long_mul(out, inq, coeff) \ + int32x4_t out##_l = vmull_s16(vget_low_s16(inq), coeff); \ + int32x4_t out##_h = vmull_s16(vget_high_s16(inq), coeff) + +# define dct_long_mac(out, acc, inq, coeff) \ + int32x4_t out##_l = vmlal_s16(acc##_l, vget_low_s16(inq), coeff); \ + int32x4_t out##_h = vmlal_s16(acc##_h, vget_high_s16(inq), coeff) + +# define dct_widen(out, inq) \ + int32x4_t out##_l = vshll_n_s16(vget_low_s16(inq), 12); \ + int32x4_t out##_h = vshll_n_s16(vget_high_s16(inq), 12) + +// wide add +# define dct_wadd(out, a, b) \ + int32x4_t out##_l = vaddq_s32(a##_l, b##_l); \ + int32x4_t out##_h = vaddq_s32(a##_h, b##_h) + +// wide sub +# define dct_wsub(out, a, b) \ + int32x4_t out##_l = vsubq_s32(a##_l, b##_l); \ + int32x4_t out##_h = vsubq_s32(a##_h, b##_h) + +// butterfly a/b, then shift using "shiftop" by "s" and pack +# define dct_bfly32o(out0, out1, a, b, shiftop, s) \ + { \ + dct_wadd(sum, a, b); \ + dct_wsub(dif, a, b); \ + out0 = vcombine_s16(shiftop(sum_l, s), shiftop(sum_h, s)); \ + out1 = vcombine_s16(shiftop(dif_l, s), shiftop(dif_h, s)); \ + } + +# define dct_pass(shiftop, shift) \ + { \ + /* even part */ \ + int16x8_t sum26 = vaddq_s16(row2, row6); \ + dct_long_mul(p1e, sum26, rot0_0); \ + dct_long_mac(t2e, p1e, row6, rot0_1); \ + dct_long_mac(t3e, p1e, row2, rot0_2); \ + int16x8_t sum04 = vaddq_s16(row0, row4); \ + int16x8_t dif04 = vsubq_s16(row0, row4); \ + dct_widen(t0e, sum04); \ + dct_widen(t1e, dif04); \ + dct_wadd(x0, t0e, t3e); \ + dct_wsub(x3, t0e, t3e); \ + dct_wadd(x1, t1e, t2e); \ + dct_wsub(x2, t1e, t2e); \ + /* odd part */ \ + int16x8_t sum15 = vaddq_s16(row1, row5); \ + int16x8_t sum17 = vaddq_s16(row1, row7); \ + int16x8_t sum35 = vaddq_s16(row3, row5); \ + int16x8_t sum37 = vaddq_s16(row3, row7); \ + int16x8_t sumodd = vaddq_s16(sum17, sum35); \ + dct_long_mul(p5o, sumodd, rot1_0); \ + dct_long_mac(p1o, p5o, sum17, rot1_1); \ + dct_long_mac(p2o, p5o, sum35, rot1_2); \ + dct_long_mul(p3o, sum37, rot2_0); \ + dct_long_mul(p4o, sum15, rot2_1); \ + dct_wadd(sump13o, p1o, p3o); \ + dct_wadd(sump24o, p2o, p4o); \ + dct_wadd(sump23o, p2o, p3o); \ + dct_wadd(sump14o, p1o, p4o); \ + dct_long_mac(x4, sump13o, row7, rot3_0); \ + dct_long_mac(x5, sump24o, row5, rot3_1); \ + dct_long_mac(x6, sump23o, row3, rot3_2); \ + dct_long_mac(x7, sump14o, row1, rot3_3); \ + dct_bfly32o(row0, row7, x0, x7, shiftop, shift); \ + dct_bfly32o(row1, row6, x1, x6, shiftop, shift); \ + dct_bfly32o(row2, row5, x2, x5, shiftop, shift); \ + dct_bfly32o(row3, row4, x3, x4, shiftop, shift); \ + } + + // load + row0 = vld1q_s16(data + 0 * 8); + row1 = vld1q_s16(data + 1 * 8); + row2 = vld1q_s16(data + 2 * 8); + row3 = vld1q_s16(data + 3 * 8); + row4 = vld1q_s16(data + 4 * 8); + row5 = vld1q_s16(data + 5 * 8); + row6 = vld1q_s16(data + 6 * 8); + row7 = vld1q_s16(data + 7 * 8); + + // add DC bias + row0 = vaddq_s16(row0, vsetq_lane_s16(1024, vdupq_n_s16(0), 0)); + + // column pass + dct_pass(vrshrn_n_s32, 10); + + // 16bit 8x8 transpose + { +// these three map to a single VTRN.16, VTRN.32, and VSWP, respectively. +// whether compilers actually get this is another story, sadly. +# define dct_trn16(x, y) \ + { \ + int16x8x2_t t = vtrnq_s16(x, y); \ + x = t.val[0]; \ + y = t.val[1]; \ + } +# define dct_trn32(x, y) \ + { \ + int32x4x2_t t = vtrnq_s32(vreinterpretq_s32_s16(x), vreinterpretq_s32_s16(y)); \ + x = vreinterpretq_s16_s32(t.val[0]); \ + y = vreinterpretq_s16_s32(t.val[1]); \ + } +# define dct_trn64(x, y) \ + { \ + int16x8_t x0 = x; \ + int16x8_t y0 = y; \ + x = vcombine_s16(vget_low_s16(x0), vget_low_s16(y0)); \ + y = vcombine_s16(vget_high_s16(x0), vget_high_s16(y0)); \ + } + + // pass 1 + dct_trn16(row0, row1); // a0b0a2b2a4b4a6b6 + dct_trn16(row2, row3); + dct_trn16(row4, row5); + dct_trn16(row6, row7); + + // pass 2 + dct_trn32(row0, row2); // a0b0c0d0a4b4c4d4 + dct_trn32(row1, row3); + dct_trn32(row4, row6); + dct_trn32(row5, row7); + + // pass 3 + dct_trn64(row0, row4); // a0b0c0d0e0f0g0h0 + dct_trn64(row1, row5); + dct_trn64(row2, row6); + dct_trn64(row3, row7); + +# undef dct_trn16 +# undef dct_trn32 +# undef dct_trn64 + } + + // row pass + // vrshrn_n_s32 only supports shifts up to 16, we need + // 17. so do a non-rounding shift of 16 first then follow + // up with a rounding shift by 1. + dct_pass(vshrn_n_s32, 16); + + { + // pack and round + uint8x8_t p0 = vqrshrun_n_s16(row0, 1); + uint8x8_t p1 = vqrshrun_n_s16(row1, 1); + uint8x8_t p2 = vqrshrun_n_s16(row2, 1); + uint8x8_t p3 = vqrshrun_n_s16(row3, 1); + uint8x8_t p4 = vqrshrun_n_s16(row4, 1); + uint8x8_t p5 = vqrshrun_n_s16(row5, 1); + uint8x8_t p6 = vqrshrun_n_s16(row6, 1); + uint8x8_t p7 = vqrshrun_n_s16(row7, 1); + + // again, these can translate into one instruction, but often don't. +# define dct_trn8_8(x, y) \ + { \ + uint8x8x2_t t = vtrn_u8(x, y); \ + x = t.val[0]; \ + y = t.val[1]; \ + } +# define dct_trn8_16(x, y) \ + { \ + uint16x4x2_t t = vtrn_u16(vreinterpret_u16_u8(x), vreinterpret_u16_u8(y)); \ + x = vreinterpret_u8_u16(t.val[0]); \ + y = vreinterpret_u8_u16(t.val[1]); \ + } +# define dct_trn8_32(x, y) \ + { \ + uint32x2x2_t t = vtrn_u32(vreinterpret_u32_u8(x), vreinterpret_u32_u8(y)); \ + x = vreinterpret_u8_u32(t.val[0]); \ + y = vreinterpret_u8_u32(t.val[1]); \ + } + + // sadly can't use interleaved stores here since we only write + // 8 bytes to each scan line! + + // 8x8 8-bit transpose pass 1 + dct_trn8_8(p0, p1); + dct_trn8_8(p2, p3); + dct_trn8_8(p4, p5); + dct_trn8_8(p6, p7); + + // pass 2 + dct_trn8_16(p0, p2); + dct_trn8_16(p1, p3); + dct_trn8_16(p4, p6); + dct_trn8_16(p5, p7); + + // pass 3 + dct_trn8_32(p0, p4); + dct_trn8_32(p1, p5); + dct_trn8_32(p2, p6); + dct_trn8_32(p3, p7); + + // store + vst1_u8(out, p0); + out += out_stride; + vst1_u8(out, p1); + out += out_stride; + vst1_u8(out, p2); + out += out_stride; + vst1_u8(out, p3); + out += out_stride; + vst1_u8(out, p4); + out += out_stride; + vst1_u8(out, p5); + out += out_stride; + vst1_u8(out, p6); + out += out_stride; + vst1_u8(out, p7); + +# undef dct_trn8_8 +# undef dct_trn8_16 +# undef dct_trn8_32 + } + +# undef dct_long_mul +# undef dct_long_mac +# undef dct_widen +# undef dct_wadd +# undef dct_wsub +# undef dct_bfly32o +# undef dct_pass +} + +# endif // STBI_NEON + +# define STBI__MARKER_none 0xff +// if there's a pending marker from the entropy stream, return that +// otherwise, fetch from the stream and get a marker. if there's no +// marker, return 0xff, which is never a valid marker value +static stbi_uc stbi__get_marker(stbi__jpeg* j) +{ + stbi_uc x; + if (j->marker != STBI__MARKER_none) { + x = j->marker; + j->marker = STBI__MARKER_none; + return x; + } + x = stbi__get8(j->s); + if (x != 0xff) + return STBI__MARKER_none; + while (x == 0xff) + x = stbi__get8(j->s); // consume repeated 0xff fill bytes + return x; +} + +// in each scan, we'll have scan_n components, and the order +// of the components is specified by order[] +# define STBI__RESTART(x) ((x) >= 0xd0 && (x) <= 0xd7) + +// after a restart interval, stbi__jpeg_reset the entropy decoder and +// the dc prediction +static void stbi__jpeg_reset(stbi__jpeg* j) +{ + j->code_bits = 0; + j->code_buffer = 0; + j->nomore = 0; + j->img_comp[0].dc_pred = j->img_comp[1].dc_pred = j->img_comp[2].dc_pred = j->img_comp[3].dc_pred = 0; + j->marker = STBI__MARKER_none; + j->todo = j->restart_interval ? j->restart_interval : 0x7fffffff; + j->eob_run = 0; + // no more than 1<<31 MCUs if no restart_interal? that's plenty safe, + // since we don't even allow 1<<30 pixels +} + +static int stbi__parse_entropy_coded_data(stbi__jpeg* z) +{ + stbi__jpeg_reset(z); + if (!z->progressive) { + if (z->scan_n == 1) { + int i, j; + STBI_SIMD_ALIGN(short, data[64]); + int n = z->order[0]; + // non-interleaved data, we just need to process one block at a time, + // in trivial scanline order + // number of blocks to do just depends on how many actual "pixels" this + // component has, independent of interleaved MCU blocking and such + int w = (z->img_comp[n].x + 7) >> 3; + int h = (z->img_comp[n].y + 7) >> 3; + for (j = 0; j < h; ++j) { + for (i = 0; i < w; ++i) { + int ha = z->img_comp[n].ha; + if (!stbi__jpeg_decode_block(z, data, z->huff_dc + z->img_comp[n].hd, z->huff_ac + ha, z->fast_ac[ha], n, z->dequant[z->img_comp[n].tq])) + return 0; + z->idct_block_kernel(z->img_comp[n].data + z->img_comp[n].w2 * j * 8 + i * 8, z->img_comp[n].w2, data); + // every data block is an MCU, so countdown the restart interval + if (--z->todo <= 0) { + if (z->code_bits < 24) + stbi__grow_buffer_unsafe(z); + // if it's NOT a restart, then just bail, so we get corrupt data + // rather than no data + if (!STBI__RESTART(z->marker)) + return 1; + stbi__jpeg_reset(z); + } + } + } + return 1; + } else { // interleaved + int i, j, k, x, y; + STBI_SIMD_ALIGN(short, data[64]); + for (j = 0; j < z->img_mcu_y; ++j) { + for (i = 0; i < z->img_mcu_x; ++i) { + // scan an interleaved mcu... process scan_n components in order + for (k = 0; k < z->scan_n; ++k) { + int n = z->order[k]; + // scan out an mcu's worth of this component; that's just determined + // by the basic H and V specified for the component + for (y = 0; y < z->img_comp[n].v; ++y) { + for (x = 0; x < z->img_comp[n].h; ++x) { + int x2 = (i * z->img_comp[n].h + x) * 8; + int y2 = (j * z->img_comp[n].v + y) * 8; + int ha = z->img_comp[n].ha; + if (!stbi__jpeg_decode_block(z, data, z->huff_dc + z->img_comp[n].hd, z->huff_ac + ha, z->fast_ac[ha], n, z->dequant[z->img_comp[n].tq])) + return 0; + z->idct_block_kernel(z->img_comp[n].data + z->img_comp[n].w2 * y2 + x2, z->img_comp[n].w2, data); + } + } + } + // after all interleaved components, that's an interleaved MCU, + // so now count down the restart interval + if (--z->todo <= 0) { + if (z->code_bits < 24) + stbi__grow_buffer_unsafe(z); + if (!STBI__RESTART(z->marker)) + return 1; + stbi__jpeg_reset(z); + } + } + } + return 1; + } + } else { + if (z->scan_n == 1) { + int i, j; + int n = z->order[0]; + // non-interleaved data, we just need to process one block at a time, + // in trivial scanline order + // number of blocks to do just depends on how many actual "pixels" this + // component has, independent of interleaved MCU blocking and such + int w = (z->img_comp[n].x + 7) >> 3; + int h = (z->img_comp[n].y + 7) >> 3; + for (j = 0; j < h; ++j) { + for (i = 0; i < w; ++i) { + short* data = z->img_comp[n].coeff + 64 * (i + j * z->img_comp[n].coeff_w); + if (z->spec_start == 0) { + if (!stbi__jpeg_decode_block_prog_dc(z, data, &z->huff_dc[z->img_comp[n].hd], n)) + return 0; + } else { + int ha = z->img_comp[n].ha; + if (!stbi__jpeg_decode_block_prog_ac(z, data, &z->huff_ac[ha], z->fast_ac[ha])) + return 0; + } + // every data block is an MCU, so countdown the restart interval + if (--z->todo <= 0) { + if (z->code_bits < 24) + stbi__grow_buffer_unsafe(z); + if (!STBI__RESTART(z->marker)) + return 1; + stbi__jpeg_reset(z); + } + } + } + return 1; + } else { // interleaved + int i, j, k, x, y; + for (j = 0; j < z->img_mcu_y; ++j) { + for (i = 0; i < z->img_mcu_x; ++i) { + // scan an interleaved mcu... process scan_n components in order + for (k = 0; k < z->scan_n; ++k) { + int n = z->order[k]; + // scan out an mcu's worth of this component; that's just determined + // by the basic H and V specified for the component + for (y = 0; y < z->img_comp[n].v; ++y) { + for (x = 0; x < z->img_comp[n].h; ++x) { + int x2 = (i * z->img_comp[n].h + x); + int y2 = (j * z->img_comp[n].v + y); + short* data = z->img_comp[n].coeff + 64 * (x2 + y2 * z->img_comp[n].coeff_w); + if (!stbi__jpeg_decode_block_prog_dc(z, data, &z->huff_dc[z->img_comp[n].hd], n)) + return 0; + } + } + } + // after all interleaved components, that's an interleaved MCU, + // so now count down the restart interval + if (--z->todo <= 0) { + if (z->code_bits < 24) + stbi__grow_buffer_unsafe(z); + if (!STBI__RESTART(z->marker)) + return 1; + stbi__jpeg_reset(z); + } + } + } + return 1; + } + } +} + +static void stbi__jpeg_dequantize(short* data, stbi__uint16* dequant) +{ + int i; + for (i = 0; i < 64; ++i) + data[i] *= dequant[i]; +} + +static void stbi__jpeg_finish(stbi__jpeg* z) +{ + if (z->progressive) { + // dequantize and idct the data + int i, j, n; + for (n = 0; n < z->s->img_n; ++n) { + int w = (z->img_comp[n].x + 7) >> 3; + int h = (z->img_comp[n].y + 7) >> 3; + for (j = 0; j < h; ++j) { + for (i = 0; i < w; ++i) { + short* data = z->img_comp[n].coeff + 64 * (i + j * z->img_comp[n].coeff_w); + stbi__jpeg_dequantize(data, z->dequant[z->img_comp[n].tq]); + z->idct_block_kernel(z->img_comp[n].data + z->img_comp[n].w2 * j * 8 + i * 8, z->img_comp[n].w2, data); + } + } + } + } +} + +static int stbi__process_marker(stbi__jpeg* z, int m) +{ + int L; + switch (m) { + case STBI__MARKER_none: // no marker found + return stbi__err("expected marker", "Corrupt JPEG"); + + case 0xDD: // DRI - specify restart interval + if (stbi__get16be(z->s) != 4) + return stbi__err("bad DRI len", "Corrupt JPEG"); + z->restart_interval = stbi__get16be(z->s); + return 1; + + case 0xDB: // DQT - define quantization table + L = stbi__get16be(z->s) - 2; + while (L > 0) { + int q = stbi__get8(z->s); + int p = q >> 4, sixteen = (p != 0); + int t = q & 15, i; + if (p != 0 && p != 1) + return stbi__err("bad DQT type", "Corrupt JPEG"); + if (t > 3) + return stbi__err("bad DQT table", "Corrupt JPEG"); + + for (i = 0; i < 64; ++i) + z->dequant[t][stbi__jpeg_dezigzag[i]] = (stbi__uint16)(sixteen ? stbi__get16be(z->s) : stbi__get8(z->s)); + L -= (sixteen ? 129 : 65); + } + return L == 0; + + case 0xC4: // DHT - define huffman table + L = stbi__get16be(z->s) - 2; + while (L > 0) { + stbi_uc* v; + int sizes[16], i, n = 0; + int q = stbi__get8(z->s); + int tc = q >> 4; + int th = q & 15; + if (tc > 1 || th > 3) + return stbi__err("bad DHT header", "Corrupt JPEG"); + for (i = 0; i < 16; ++i) { + sizes[i] = stbi__get8(z->s); + n += sizes[i]; + } + if (n > 256) + return stbi__err("bad DHT header", "Corrupt JPEG"); // Loop over i < n would write past end of values! + L -= 17; + if (tc == 0) { + if (!stbi__build_huffman(z->huff_dc + th, sizes)) + return 0; + v = z->huff_dc[th].values; + } else { + if (!stbi__build_huffman(z->huff_ac + th, sizes)) + return 0; + v = z->huff_ac[th].values; + } + for (i = 0; i < n; ++i) + v[i] = stbi__get8(z->s); + if (tc != 0) + stbi__build_fast_ac(z->fast_ac[th], z->huff_ac + th); + L -= n; + } + return L == 0; + } + + // check for comment block or APP blocks + if ((m >= 0xE0 && m <= 0xEF) || m == 0xFE) { + L = stbi__get16be(z->s); + if (L < 2) { + if (m == 0xFE) + return stbi__err("bad COM len", "Corrupt JPEG"); + else + return stbi__err("bad APP len", "Corrupt JPEG"); + } + L -= 2; + + if (m == 0xE0 && L >= 5) { // JFIF APP0 segment + static unsigned char const tag[5] = { 'J', 'F', 'I', 'F', '\0' }; + int ok = 1; + int i; + for (i = 0; i < 5; ++i) + if (stbi__get8(z->s) != tag[i]) + ok = 0; + L -= 5; + if (ok) + z->jfif = 1; + } else if (m == 0xEE && L >= 12) { // Adobe APP14 segment + static unsigned char const tag[6] = { 'A', 'd', 'o', 'b', 'e', '\0' }; + int ok = 1; + int i; + for (i = 0; i < 6; ++i) + if (stbi__get8(z->s) != tag[i]) + ok = 0; + L -= 6; + if (ok) { + stbi__get8(z->s); // version + stbi__get16be(z->s); // flags0 + stbi__get16be(z->s); // flags1 + z->app14_color_transform = stbi__get8(z->s); // color transform + L -= 6; + } + } + + stbi__skip(z->s, L); + return 1; + } + + return stbi__err("unknown marker", "Corrupt JPEG"); +} + +// after we see SOS +static int stbi__process_scan_header(stbi__jpeg* z) +{ + int i; + int Ls = stbi__get16be(z->s); + z->scan_n = stbi__get8(z->s); + if (z->scan_n < 1 || z->scan_n > 4 || z->scan_n > (int)z->s->img_n) + return stbi__err("bad SOS component count", "Corrupt JPEG"); + if (Ls != 6 + 2 * z->scan_n) + return stbi__err("bad SOS len", "Corrupt JPEG"); + for (i = 0; i < z->scan_n; ++i) { + int id = stbi__get8(z->s), which; + int q = stbi__get8(z->s); + for (which = 0; which < z->s->img_n; ++which) + if (z->img_comp[which].id == id) + break; + if (which == z->s->img_n) + return 0; // no match + z->img_comp[which].hd = q >> 4; + if (z->img_comp[which].hd > 3) + return stbi__err("bad DC huff", "Corrupt JPEG"); + z->img_comp[which].ha = q & 15; + if (z->img_comp[which].ha > 3) + return stbi__err("bad AC huff", "Corrupt JPEG"); + z->order[i] = which; + } + + { + int aa; + z->spec_start = stbi__get8(z->s); + z->spec_end = stbi__get8(z->s); // should be 63, but might be 0 + aa = stbi__get8(z->s); + z->succ_high = (aa >> 4); + z->succ_low = (aa & 15); + if (z->progressive) { + if (z->spec_start > 63 || z->spec_end > 63 || z->spec_start > z->spec_end || z->succ_high > 13 || z->succ_low > 13) + return stbi__err("bad SOS", "Corrupt JPEG"); + } else { + if (z->spec_start != 0) + return stbi__err("bad SOS", "Corrupt JPEG"); + if (z->succ_high != 0 || z->succ_low != 0) + return stbi__err("bad SOS", "Corrupt JPEG"); + z->spec_end = 63; + } + } + + return 1; +} + +static int stbi__free_jpeg_components(stbi__jpeg* z, int ncomp, int why) +{ + int i; + for (i = 0; i < ncomp; ++i) { + if (z->img_comp[i].raw_data) { + STBI_FREE(z->img_comp[i].raw_data); + z->img_comp[i].raw_data = NULL; + z->img_comp[i].data = NULL; + } + if (z->img_comp[i].raw_coeff) { + STBI_FREE(z->img_comp[i].raw_coeff); + z->img_comp[i].raw_coeff = 0; + z->img_comp[i].coeff = 0; + } + if (z->img_comp[i].linebuf) { + STBI_FREE(z->img_comp[i].linebuf); + z->img_comp[i].linebuf = NULL; + } + } + return why; +} + +static int stbi__process_frame_header(stbi__jpeg* z, int scan) +{ + stbi__context* s = z->s; + int Lf, p, i, q, h_max = 1, v_max = 1, c; + Lf = stbi__get16be(s); + if (Lf < 11) + return stbi__err("bad SOF len", "Corrupt JPEG"); // JPEG + p = stbi__get8(s); + if (p != 8) + return stbi__err("only 8-bit", "JPEG format not supported: 8-bit only"); // JPEG baseline + s->img_y = stbi__get16be(s); + if (s->img_y == 0) + return stbi__err("no header height", "JPEG format not supported: delayed height"); // Legal, but we don't handle it--but neither does IJG + s->img_x = stbi__get16be(s); + if (s->img_x == 0) + return stbi__err("0 width", "Corrupt JPEG"); // JPEG requires + if (s->img_y > STBI_MAX_DIMENSIONS) + return stbi__err("too large", "Very large image (corrupt?)"); + if (s->img_x > STBI_MAX_DIMENSIONS) + return stbi__err("too large", "Very large image (corrupt?)"); + c = stbi__get8(s); + if (c != 3 && c != 1 && c != 4) + return stbi__err("bad component count", "Corrupt JPEG"); + s->img_n = c; + for (i = 0; i < c; ++i) { + z->img_comp[i].data = NULL; + z->img_comp[i].linebuf = NULL; + } + + if (Lf != 8 + 3 * s->img_n) + return stbi__err("bad SOF len", "Corrupt JPEG"); + + z->rgb = 0; + for (i = 0; i < s->img_n; ++i) { + static unsigned char const rgb[3] = { 'R', 'G', 'B' }; + z->img_comp[i].id = stbi__get8(s); + if (s->img_n == 3 && z->img_comp[i].id == rgb[i]) + ++z->rgb; + q = stbi__get8(s); + z->img_comp[i].h = (q >> 4); + if (!z->img_comp[i].h || z->img_comp[i].h > 4) + return stbi__err("bad H", "Corrupt JPEG"); + z->img_comp[i].v = q & 15; + if (!z->img_comp[i].v || z->img_comp[i].v > 4) + return stbi__err("bad V", "Corrupt JPEG"); + z->img_comp[i].tq = stbi__get8(s); + if (z->img_comp[i].tq > 3) + return stbi__err("bad TQ", "Corrupt JPEG"); + } + + if (scan != STBI__SCAN_load) + return 1; + + if (!stbi__mad3sizes_valid(s->img_x, s->img_y, s->img_n, 0)) + return stbi__err("too large", "Image too large to decode"); + + for (i = 0; i < s->img_n; ++i) { + if (z->img_comp[i].h > h_max) + h_max = z->img_comp[i].h; + if (z->img_comp[i].v > v_max) + v_max = z->img_comp[i].v; + } + + // check that plane subsampling factors are integer ratios; our resamplers can't deal with fractional ratios + // and I've never seen a non-corrupted JPEG file actually use them + for (i = 0; i < s->img_n; ++i) { + if (h_max % z->img_comp[i].h != 0) + return stbi__err("bad H", "Corrupt JPEG"); + if (v_max % z->img_comp[i].v != 0) + return stbi__err("bad V", "Corrupt JPEG"); + } + + // compute interleaved mcu info + z->img_h_max = h_max; + z->img_v_max = v_max; + z->img_mcu_w = h_max * 8; + z->img_mcu_h = v_max * 8; + // these sizes can't be more than 17 bits + z->img_mcu_x = (s->img_x + z->img_mcu_w - 1) / z->img_mcu_w; + z->img_mcu_y = (s->img_y + z->img_mcu_h - 1) / z->img_mcu_h; + + for (i = 0; i < s->img_n; ++i) { + // number of effective pixels (e.g. for non-interleaved MCU) + z->img_comp[i].x = (s->img_x * z->img_comp[i].h + h_max - 1) / h_max; + z->img_comp[i].y = (s->img_y * z->img_comp[i].v + v_max - 1) / v_max; + // to simplify generation, we'll allocate enough memory to decode + // the bogus oversized data from using interleaved MCUs and their + // big blocks (e.g. a 16x16 iMCU on an image of width 33); we won't + // discard the extra data until colorspace conversion + // + // img_mcu_x, img_mcu_y: <=17 bits; comp[i].h and .v are <=4 (checked earlier) + // so these muls can't overflow with 32-bit ints (which we require) + z->img_comp[i].w2 = z->img_mcu_x * z->img_comp[i].h * 8; + z->img_comp[i].h2 = z->img_mcu_y * z->img_comp[i].v * 8; + z->img_comp[i].coeff = 0; + z->img_comp[i].raw_coeff = 0; + z->img_comp[i].linebuf = NULL; + z->img_comp[i].raw_data = stbi__malloc_mad2(z->img_comp[i].w2, z->img_comp[i].h2, 15); + if (z->img_comp[i].raw_data == NULL) + return stbi__free_jpeg_components(z, i + 1, stbi__err("outofmem", "Out of memory")); + // align blocks for idct using mmx/sse + z->img_comp[i].data = (stbi_uc*)(((size_t)z->img_comp[i].raw_data + 15) & ~15); + if (z->progressive) { + // w2, h2 are multiples of 8 (see above) + z->img_comp[i].coeff_w = z->img_comp[i].w2 / 8; + z->img_comp[i].coeff_h = z->img_comp[i].h2 / 8; + z->img_comp[i].raw_coeff = stbi__malloc_mad3(z->img_comp[i].w2, z->img_comp[i].h2, sizeof(short), 15); + if (z->img_comp[i].raw_coeff == NULL) + return stbi__free_jpeg_components(z, i + 1, stbi__err("outofmem", "Out of memory")); + z->img_comp[i].coeff = (short*)(((size_t)z->img_comp[i].raw_coeff + 15) & ~15); + } + } + + return 1; +} + +// use comparisons since in some cases we handle more than one case (e.g. SOF) +# define stbi__DNL(x) ((x) == 0xdc) +# define stbi__SOI(x) ((x) == 0xd8) +# define stbi__EOI(x) ((x) == 0xd9) +# define stbi__SOF(x) ((x) == 0xc0 || (x) == 0xc1 || (x) == 0xc2) +# define stbi__SOS(x) ((x) == 0xda) + +# define stbi__SOF_progressive(x) ((x) == 0xc2) + +static int stbi__decode_jpeg_header(stbi__jpeg* z, int scan) +{ + int m; + z->jfif = 0; + z->app14_color_transform = -1; // valid values are 0,1,2 + z->marker = STBI__MARKER_none; // initialize cached marker to empty + m = stbi__get_marker(z); + if (!stbi__SOI(m)) + return stbi__err("no SOI", "Corrupt JPEG"); + if (scan == STBI__SCAN_type) + return 1; + m = stbi__get_marker(z); + while (!stbi__SOF(m)) { + if (!stbi__process_marker(z, m)) + return 0; + m = stbi__get_marker(z); + while (m == STBI__MARKER_none) { + // some files have extra padding after their blocks, so ok, we'll scan + if (stbi__at_eof(z->s)) + return stbi__err("no SOF", "Corrupt JPEG"); + m = stbi__get_marker(z); + } + } + z->progressive = stbi__SOF_progressive(m); + if (!stbi__process_frame_header(z, scan)) + return 0; + return 1; +} + +static stbi_uc stbi__skip_jpeg_junk_at_end(stbi__jpeg* j) +{ + // some JPEGs have junk at end, skip over it but if we find what looks + // like a valid marker, resume there + while (!stbi__at_eof(j->s)) { + stbi_uc x = stbi__get8(j->s); + while (x == 0xff) { // might be a marker + if (stbi__at_eof(j->s)) + return STBI__MARKER_none; + x = stbi__get8(j->s); + if (x != 0x00 && x != 0xff) { + // not a stuffed zero or lead-in to another marker, looks + // like an actual marker, return it + return x; + } + // stuffed zero has x=0 now which ends the loop, meaning we go + // back to regular scan loop. + // repeated 0xff keeps trying to read the next byte of the marker. + } + } + return STBI__MARKER_none; +} + +// decode image to YCbCr format +static int stbi__decode_jpeg_image(stbi__jpeg* j) +{ + int m; + for (m = 0; m < 4; m++) { + j->img_comp[m].raw_data = NULL; + j->img_comp[m].raw_coeff = NULL; + } + j->restart_interval = 0; + if (!stbi__decode_jpeg_header(j, STBI__SCAN_load)) + return 0; + m = stbi__get_marker(j); + while (!stbi__EOI(m)) { + if (stbi__SOS(m)) { + if (!stbi__process_scan_header(j)) + return 0; + if (!stbi__parse_entropy_coded_data(j)) + return 0; + if (j->marker == STBI__MARKER_none) { + j->marker = stbi__skip_jpeg_junk_at_end(j); + // if we reach eof without hitting a marker, stbi__get_marker() below will fail and we'll eventually return 0 + } + m = stbi__get_marker(j); + if (STBI__RESTART(m)) + m = stbi__get_marker(j); + } else if (stbi__DNL(m)) { + int Ld = stbi__get16be(j->s); + stbi__uint32 NL = stbi__get16be(j->s); + if (Ld != 4) + return stbi__err("bad DNL len", "Corrupt JPEG"); + if (NL != j->s->img_y) + return stbi__err("bad DNL height", "Corrupt JPEG"); + m = stbi__get_marker(j); + } else { + if (!stbi__process_marker(j, m)) + return 1; + m = stbi__get_marker(j); + } + } + if (j->progressive) + stbi__jpeg_finish(j); + return 1; +} + +// static jfif-centered resampling (across block boundaries) + +typedef stbi_uc* (*resample_row_func)(stbi_uc* out, stbi_uc* in0, stbi_uc* in1, + int w, int hs); + +# define stbi__div4(x) ((stbi_uc)((x) >> 2)) + +static stbi_uc* resample_row_1(stbi_uc* out, stbi_uc* in_near, stbi_uc* in_far, int w, int hs) +{ + STBI_NOTUSED(out); + STBI_NOTUSED(in_far); + STBI_NOTUSED(w); + STBI_NOTUSED(hs); + return in_near; +} + +static stbi_uc* stbi__resample_row_v_2(stbi_uc* out, stbi_uc* in_near, stbi_uc* in_far, int w, int hs) +{ + // need to generate two samples vertically for every one in input + int i; + STBI_NOTUSED(hs); + for (i = 0; i < w; ++i) + out[i] = stbi__div4(3 * in_near[i] + in_far[i] + 2); + return out; +} + +static stbi_uc* stbi__resample_row_h_2(stbi_uc* out, stbi_uc* in_near, stbi_uc* in_far, int w, int hs) +{ + // need to generate two samples horizontally for every one in input + int i; + stbi_uc* input = in_near; + + if (w == 1) { + // if only one sample, can't do any interpolation + out[0] = out[1] = input[0]; + return out; + } + + out[0] = input[0]; + out[1] = stbi__div4(input[0] * 3 + input[1] + 2); + for (i = 1; i < w - 1; ++i) { + int n = 3 * input[i] + 2; + out[i * 2 + 0] = stbi__div4(n + input[i - 1]); + out[i * 2 + 1] = stbi__div4(n + input[i + 1]); + } + out[i * 2 + 0] = stbi__div4(input[w - 2] * 3 + input[w - 1] + 2); + out[i * 2 + 1] = input[w - 1]; + + STBI_NOTUSED(in_far); + STBI_NOTUSED(hs); + + return out; +} + +# define stbi__div16(x) ((stbi_uc)((x) >> 4)) + +static stbi_uc* stbi__resample_row_hv_2(stbi_uc* out, stbi_uc* in_near, stbi_uc* in_far, int w, int hs) +{ + // need to generate 2x2 samples for every one in input + int i, t0, t1; + if (w == 1) { + out[0] = out[1] = stbi__div4(3 * in_near[0] + in_far[0] + 2); + return out; + } + + t1 = 3 * in_near[0] + in_far[0]; + out[0] = stbi__div4(t1 + 2); + for (i = 1; i < w; ++i) { + t0 = t1; + t1 = 3 * in_near[i] + in_far[i]; + out[i * 2 - 1] = stbi__div16(3 * t0 + t1 + 8); + out[i * 2] = stbi__div16(3 * t1 + t0 + 8); + } + out[w * 2 - 1] = stbi__div4(t1 + 2); + + STBI_NOTUSED(hs); + + return out; +} + +# if defined(STBI_SSE2) || defined(STBI_NEON) +static stbi_uc* stbi__resample_row_hv_2_simd(stbi_uc* out, stbi_uc* in_near, stbi_uc* in_far, int w, int hs) +{ + // need to generate 2x2 samples for every one in input + int i = 0, t0, t1; + + if (w == 1) { + out[0] = out[1] = stbi__div4(3 * in_near[0] + in_far[0] + 2); + return out; + } + + t1 = 3 * in_near[0] + in_far[0]; + // process groups of 8 pixels for as long as we can. + // note we can't handle the last pixel in a row in this loop + // because we need to handle the filter boundary conditions. + for (; i < ((w - 1) & ~7); i += 8) { +# if defined(STBI_SSE2) + // load and perform the vertical filtering pass + // this uses 3*x + y = 4*x + (y - x) + __m128i zero = _mm_setzero_si128(); + __m128i farb = _mm_loadl_epi64((__m128i*)(in_far + i)); + __m128i nearb = _mm_loadl_epi64((__m128i*)(in_near + i)); + __m128i farw = _mm_unpacklo_epi8(farb, zero); + __m128i nearw = _mm_unpacklo_epi8(nearb, zero); + __m128i diff = _mm_sub_epi16(farw, nearw); + __m128i nears = _mm_slli_epi16(nearw, 2); + __m128i curr = _mm_add_epi16(nears, diff); // current row + + // horizontal filter works the same based on shifted vers of current + // row. "prev" is current row shifted right by 1 pixel; we need to + // insert the previous pixel value (from t1). + // "next" is current row shifted left by 1 pixel, with first pixel + // of next block of 8 pixels added in. + __m128i prv0 = _mm_slli_si128(curr, 2); + __m128i nxt0 = _mm_srli_si128(curr, 2); + __m128i prev = _mm_insert_epi16(prv0, t1, 0); + __m128i next = _mm_insert_epi16(nxt0, 3 * in_near[i + 8] + in_far[i + 8], 7); + + // horizontal filter, polyphase implementation since it's convenient: + // even pixels = 3*cur + prev = cur*4 + (prev - cur) + // odd pixels = 3*cur + next = cur*4 + (next - cur) + // note the shared term. + __m128i bias = _mm_set1_epi16(8); + __m128i curs = _mm_slli_epi16(curr, 2); + __m128i prvd = _mm_sub_epi16(prev, curr); + __m128i nxtd = _mm_sub_epi16(next, curr); + __m128i curb = _mm_add_epi16(curs, bias); + __m128i even = _mm_add_epi16(prvd, curb); + __m128i odd = _mm_add_epi16(nxtd, curb); + + // interleave even and odd pixels, then undo scaling. + __m128i int0 = _mm_unpacklo_epi16(even, odd); + __m128i int1 = _mm_unpackhi_epi16(even, odd); + __m128i de0 = _mm_srli_epi16(int0, 4); + __m128i de1 = _mm_srli_epi16(int1, 4); + + // pack and write output + __m128i outv = _mm_packus_epi16(de0, de1); + _mm_storeu_si128((__m128i*)(out + i * 2), outv); +# elif defined(STBI_NEON) + // load and perform the vertical filtering pass + // this uses 3*x + y = 4*x + (y - x) + uint8x8_t farb = vld1_u8(in_far + i); + uint8x8_t nearb = vld1_u8(in_near + i); + int16x8_t diff = vreinterpretq_s16_u16(vsubl_u8(farb, nearb)); + int16x8_t nears = vreinterpretq_s16_u16(vshll_n_u8(nearb, 2)); + int16x8_t curr = vaddq_s16(nears, diff); // current row + + // horizontal filter works the same based on shifted vers of current + // row. "prev" is current row shifted right by 1 pixel; we need to + // insert the previous pixel value (from t1). + // "next" is current row shifted left by 1 pixel, with first pixel + // of next block of 8 pixels added in. + int16x8_t prv0 = vextq_s16(curr, curr, 7); + int16x8_t nxt0 = vextq_s16(curr, curr, 1); + int16x8_t prev = vsetq_lane_s16(t1, prv0, 0); + int16x8_t next = vsetq_lane_s16(3 * in_near[i + 8] + in_far[i + 8], nxt0, 7); + + // horizontal filter, polyphase implementation since it's convenient: + // even pixels = 3*cur + prev = cur*4 + (prev - cur) + // odd pixels = 3*cur + next = cur*4 + (next - cur) + // note the shared term. + int16x8_t curs = vshlq_n_s16(curr, 2); + int16x8_t prvd = vsubq_s16(prev, curr); + int16x8_t nxtd = vsubq_s16(next, curr); + int16x8_t even = vaddq_s16(curs, prvd); + int16x8_t odd = vaddq_s16(curs, nxtd); + + // undo scaling and round, then store with even/odd phases interleaved + uint8x8x2_t o; + o.val[0] = vqrshrun_n_s16(even, 4); + o.val[1] = vqrshrun_n_s16(odd, 4); + vst2_u8(out + i * 2, o); +# endif + + // "previous" value for next iter + t1 = 3 * in_near[i + 7] + in_far[i + 7]; + } + + t0 = t1; + t1 = 3 * in_near[i] + in_far[i]; + out[i * 2] = stbi__div16(3 * t1 + t0 + 8); + + for (++i; i < w; ++i) { + t0 = t1; + t1 = 3 * in_near[i] + in_far[i]; + out[i * 2 - 1] = stbi__div16(3 * t0 + t1 + 8); + out[i * 2] = stbi__div16(3 * t1 + t0 + 8); + } + out[w * 2 - 1] = stbi__div4(t1 + 2); + + STBI_NOTUSED(hs); + + return out; +} +# endif + +static stbi_uc* stbi__resample_row_generic(stbi_uc* out, stbi_uc* in_near, stbi_uc* in_far, int w, int hs) +{ + // resample with nearest-neighbor + int i, j; + STBI_NOTUSED(in_far); + for (i = 0; i < w; ++i) + for (j = 0; j < hs; ++j) + out[i * hs + j] = in_near[i]; + return out; +} + +// this is a reduced-precision calculation of YCbCr-to-RGB introduced +// to make sure the code produces the same results in both SIMD and scalar +# define stbi__float2fixed(x) (((int)((x) * 4096.0f + 0.5f)) << 8) +static void stbi__YCbCr_to_RGB_row(stbi_uc* out, stbi_uc const* y, stbi_uc const* pcb, stbi_uc const* pcr, int count, int step) +{ + int i; + for (i = 0; i < count; ++i) { + int y_fixed = (y[i] << 20) + (1 << 19); // rounding + int r, g, b; + int cr = pcr[i] - 128; + int cb = pcb[i] - 128; + r = y_fixed + cr * stbi__float2fixed(1.40200f); + g = y_fixed + (cr * -stbi__float2fixed(0.71414f)) + ((cb * -stbi__float2fixed(0.34414f)) & 0xffff0000); + b = y_fixed + cb * stbi__float2fixed(1.77200f); + r >>= 20; + g >>= 20; + b >>= 20; + if ((unsigned)r > 255) { + if (r < 0) + r = 0; + else + r = 255; + } + if ((unsigned)g > 255) { + if (g < 0) + g = 0; + else + g = 255; + } + if ((unsigned)b > 255) { + if (b < 0) + b = 0; + else + b = 255; + } + out[0] = (stbi_uc)r; + out[1] = (stbi_uc)g; + out[2] = (stbi_uc)b; + out[3] = 255; + out += step; + } +} + +# if defined(STBI_SSE2) || defined(STBI_NEON) +static void stbi__YCbCr_to_RGB_simd(stbi_uc* out, stbi_uc const* y, stbi_uc const* pcb, stbi_uc const* pcr, int count, int step) +{ + int i = 0; + +# ifdef STBI_SSE2 + // step == 3 is pretty ugly on the final interleave, and i'm not convinced + // it's useful in practice (you wouldn't use it for textures, for example). + // so just accelerate step == 4 case. + if (step == 4) { + // this is a fairly straightforward implementation and not super-optimized. + __m128i signflip = _mm_set1_epi8(-0x80); + __m128i cr_const0 = _mm_set1_epi16((short)(1.40200f * 4096.0f + 0.5f)); + __m128i cr_const1 = _mm_set1_epi16(-(short)(0.71414f * 4096.0f + 0.5f)); + __m128i cb_const0 = _mm_set1_epi16(-(short)(0.34414f * 4096.0f + 0.5f)); + __m128i cb_const1 = _mm_set1_epi16((short)(1.77200f * 4096.0f + 0.5f)); + __m128i y_bias = _mm_set1_epi8((char)(unsigned char)128); + __m128i xw = _mm_set1_epi16(255); // alpha channel + + for (; i + 7 < count; i += 8) { + // load + __m128i y_bytes = _mm_loadl_epi64((__m128i*)(y + i)); + __m128i cr_bytes = _mm_loadl_epi64((__m128i*)(pcr + i)); + __m128i cb_bytes = _mm_loadl_epi64((__m128i*)(pcb + i)); + __m128i cr_biased = _mm_xor_si128(cr_bytes, signflip); // -128 + __m128i cb_biased = _mm_xor_si128(cb_bytes, signflip); // -128 + + // unpack to short (and left-shift cr, cb by 8) + __m128i yw = _mm_unpacklo_epi8(y_bias, y_bytes); + __m128i crw = _mm_unpacklo_epi8(_mm_setzero_si128(), cr_biased); + __m128i cbw = _mm_unpacklo_epi8(_mm_setzero_si128(), cb_biased); + + // color transform + __m128i yws = _mm_srli_epi16(yw, 4); + __m128i cr0 = _mm_mulhi_epi16(cr_const0, crw); + __m128i cb0 = _mm_mulhi_epi16(cb_const0, cbw); + __m128i cb1 = _mm_mulhi_epi16(cbw, cb_const1); + __m128i cr1 = _mm_mulhi_epi16(crw, cr_const1); + __m128i rws = _mm_add_epi16(cr0, yws); + __m128i gwt = _mm_add_epi16(cb0, yws); + __m128i bws = _mm_add_epi16(yws, cb1); + __m128i gws = _mm_add_epi16(gwt, cr1); + + // descale + __m128i rw = _mm_srai_epi16(rws, 4); + __m128i bw = _mm_srai_epi16(bws, 4); + __m128i gw = _mm_srai_epi16(gws, 4); + + // back to byte, set up for transpose + __m128i brb = _mm_packus_epi16(rw, bw); + __m128i gxb = _mm_packus_epi16(gw, xw); + + // transpose to interleave channels + __m128i t0 = _mm_unpacklo_epi8(brb, gxb); + __m128i t1 = _mm_unpackhi_epi8(brb, gxb); + __m128i o0 = _mm_unpacklo_epi16(t0, t1); + __m128i o1 = _mm_unpackhi_epi16(t0, t1); + + // store + _mm_storeu_si128((__m128i*)(out + 0), o0); + _mm_storeu_si128((__m128i*)(out + 16), o1); + out += 32; + } + } +# endif + +# ifdef STBI_NEON + // in this version, step=3 support would be easy to add. but is there demand? + if (step == 4) { + // this is a fairly straightforward implementation and not super-optimized. + uint8x8_t signflip = vdup_n_u8(0x80); + int16x8_t cr_const0 = vdupq_n_s16((short)(1.40200f * 4096.0f + 0.5f)); + int16x8_t cr_const1 = vdupq_n_s16(-(short)(0.71414f * 4096.0f + 0.5f)); + int16x8_t cb_const0 = vdupq_n_s16(-(short)(0.34414f * 4096.0f + 0.5f)); + int16x8_t cb_const1 = vdupq_n_s16((short)(1.77200f * 4096.0f + 0.5f)); + + for (; i + 7 < count; i += 8) { + // load + uint8x8_t y_bytes = vld1_u8(y + i); + uint8x8_t cr_bytes = vld1_u8(pcr + i); + uint8x8_t cb_bytes = vld1_u8(pcb + i); + int8x8_t cr_biased = vreinterpret_s8_u8(vsub_u8(cr_bytes, signflip)); + int8x8_t cb_biased = vreinterpret_s8_u8(vsub_u8(cb_bytes, signflip)); + + // expand to s16 + int16x8_t yws = vreinterpretq_s16_u16(vshll_n_u8(y_bytes, 4)); + int16x8_t crw = vshll_n_s8(cr_biased, 7); + int16x8_t cbw = vshll_n_s8(cb_biased, 7); + + // color transform + int16x8_t cr0 = vqdmulhq_s16(crw, cr_const0); + int16x8_t cb0 = vqdmulhq_s16(cbw, cb_const0); + int16x8_t cr1 = vqdmulhq_s16(crw, cr_const1); + int16x8_t cb1 = vqdmulhq_s16(cbw, cb_const1); + int16x8_t rws = vaddq_s16(yws, cr0); + int16x8_t gws = vaddq_s16(vaddq_s16(yws, cb0), cr1); + int16x8_t bws = vaddq_s16(yws, cb1); + + // undo scaling, round, convert to byte + uint8x8x4_t o; + o.val[0] = vqrshrun_n_s16(rws, 4); + o.val[1] = vqrshrun_n_s16(gws, 4); + o.val[2] = vqrshrun_n_s16(bws, 4); + o.val[3] = vdup_n_u8(255); + + // store, interleaving r/g/b/a + vst4_u8(out, o); + out += 8 * 4; + } + } +# endif + + for (; i < count; ++i) { + int y_fixed = (y[i] << 20) + (1 << 19); // rounding + int r, g, b; + int cr = pcr[i] - 128; + int cb = pcb[i] - 128; + r = y_fixed + cr * stbi__float2fixed(1.40200f); + g = y_fixed + cr * -stbi__float2fixed(0.71414f) + ((cb * -stbi__float2fixed(0.34414f)) & 0xffff0000); + b = y_fixed + cb * stbi__float2fixed(1.77200f); + r >>= 20; + g >>= 20; + b >>= 20; + if ((unsigned)r > 255) { + if (r < 0) + r = 0; + else + r = 255; + } + if ((unsigned)g > 255) { + if (g < 0) + g = 0; + else + g = 255; + } + if ((unsigned)b > 255) { + if (b < 0) + b = 0; + else + b = 255; + } + out[0] = (stbi_uc)r; + out[1] = (stbi_uc)g; + out[2] = (stbi_uc)b; + out[3] = 255; + out += step; + } +} +# endif + +// set up the kernels +static void stbi__setup_jpeg(stbi__jpeg* j) +{ + j->idct_block_kernel = stbi__idct_block; + j->YCbCr_to_RGB_kernel = stbi__YCbCr_to_RGB_row; + j->resample_row_hv_2_kernel = stbi__resample_row_hv_2; + +# ifdef STBI_SSE2 + if (stbi__sse2_available()) { + j->idct_block_kernel = stbi__idct_simd; + j->YCbCr_to_RGB_kernel = stbi__YCbCr_to_RGB_simd; + j->resample_row_hv_2_kernel = stbi__resample_row_hv_2_simd; + } +# endif + +# ifdef STBI_NEON + j->idct_block_kernel = stbi__idct_simd; + j->YCbCr_to_RGB_kernel = stbi__YCbCr_to_RGB_simd; + j->resample_row_hv_2_kernel = stbi__resample_row_hv_2_simd; +# endif +} + +// clean up the temporary component buffers +static void stbi__cleanup_jpeg(stbi__jpeg* j) +{ + stbi__free_jpeg_components(j, j->s->img_n, 0); +} + +typedef struct +{ + resample_row_func resample; + stbi_uc *line0, *line1; + int hs, vs; // expansion factor in each axis + int w_lores; // horizontal pixels pre-expansion + int ystep; // how far through vertical expansion we are + int ypos; // which pre-expansion row we're on +} stbi__resample; + +// fast 0..255 * 0..255 => 0..255 rounded multiplication +static stbi_uc stbi__blinn_8x8(stbi_uc x, stbi_uc y) +{ + unsigned int t = x * y + 128; + return (stbi_uc)((t + (t >> 8)) >> 8); +} + +static stbi_uc* load_jpeg_image(stbi__jpeg* z, int* out_x, int* out_y, int* comp, int req_comp) +{ + int n, decode_n, is_rgb; + z->s->img_n = 0; // make stbi__cleanup_jpeg safe + + // validate req_comp + if (req_comp < 0 || req_comp > 4) + return stbi__errpuc("bad req_comp", "Internal error"); + + // load a jpeg image from whichever source, but leave in YCbCr format + if (!stbi__decode_jpeg_image(z)) { + stbi__cleanup_jpeg(z); + return NULL; + } + + // determine actual number of components to generate + n = req_comp ? req_comp : z->s->img_n >= 3 ? 3 + : 1; + + is_rgb = z->s->img_n == 3 && (z->rgb == 3 || (z->app14_color_transform == 0 && !z->jfif)); + + if (z->s->img_n == 3 && n < 3 && !is_rgb) + decode_n = 1; + else + decode_n = z->s->img_n; + + // nothing to do if no components requested; check this now to avoid + // accessing uninitialized coutput[0] later + if (decode_n <= 0) { + stbi__cleanup_jpeg(z); + return NULL; + } + + // resample and color-convert + { + int k; + unsigned int i, j; + stbi_uc* output; + stbi_uc* coutput[4] = { NULL, NULL, NULL, NULL }; + + stbi__resample res_comp[4]; + + for (k = 0; k < decode_n; ++k) { + stbi__resample* r = &res_comp[k]; + + // allocate line buffer big enough for upsampling off the edges + // with upsample factor of 4 + z->img_comp[k].linebuf = (stbi_uc*)stbi__malloc(z->s->img_x + 3); + if (!z->img_comp[k].linebuf) { + stbi__cleanup_jpeg(z); + return stbi__errpuc("outofmem", "Out of memory"); + } + + r->hs = z->img_h_max / z->img_comp[k].h; + r->vs = z->img_v_max / z->img_comp[k].v; + r->ystep = r->vs >> 1; + r->w_lores = (z->s->img_x + r->hs - 1) / r->hs; + r->ypos = 0; + r->line0 = r->line1 = z->img_comp[k].data; + + if (r->hs == 1 && r->vs == 1) + r->resample = resample_row_1; + else if (r->hs == 1 && r->vs == 2) + r->resample = stbi__resample_row_v_2; + else if (r->hs == 2 && r->vs == 1) + r->resample = stbi__resample_row_h_2; + else if (r->hs == 2 && r->vs == 2) + r->resample = z->resample_row_hv_2_kernel; + else + r->resample = stbi__resample_row_generic; + } + + // can't error after this so, this is safe + output = (stbi_uc*)stbi__malloc_mad3(n, z->s->img_x, z->s->img_y, 1); + if (!output) { + stbi__cleanup_jpeg(z); + return stbi__errpuc("outofmem", "Out of memory"); + } + + // now go ahead and resample + for (j = 0; j < z->s->img_y; ++j) { + stbi_uc* out = output + n * z->s->img_x * j; + for (k = 0; k < decode_n; ++k) { + stbi__resample* r = &res_comp[k]; + int y_bot = r->ystep >= (r->vs >> 1); + coutput[k] = r->resample(z->img_comp[k].linebuf, + y_bot ? r->line1 : r->line0, + y_bot ? r->line0 : r->line1, + r->w_lores, r->hs); + if (++r->ystep >= r->vs) { + r->ystep = 0; + r->line0 = r->line1; + if (++r->ypos < z->img_comp[k].y) + r->line1 += z->img_comp[k].w2; + } + } + if (n >= 3) { + stbi_uc* y = coutput[0]; + if (z->s->img_n == 3) { + if (is_rgb) { + for (i = 0; i < z->s->img_x; ++i) { + out[0] = y[i]; + out[1] = coutput[1][i]; + out[2] = coutput[2][i]; + out[3] = 255; + out += n; + } + } else { + z->YCbCr_to_RGB_kernel(out, y, coutput[1], coutput[2], z->s->img_x, n); + } + } else if (z->s->img_n == 4) { + if (z->app14_color_transform == 0) { // CMYK + for (i = 0; i < z->s->img_x; ++i) { + stbi_uc m = coutput[3][i]; + out[0] = stbi__blinn_8x8(coutput[0][i], m); + out[1] = stbi__blinn_8x8(coutput[1][i], m); + out[2] = stbi__blinn_8x8(coutput[2][i], m); + out[3] = 255; + out += n; + } + } else if (z->app14_color_transform == 2) { // YCCK + z->YCbCr_to_RGB_kernel(out, y, coutput[1], coutput[2], z->s->img_x, n); + for (i = 0; i < z->s->img_x; ++i) { + stbi_uc m = coutput[3][i]; + out[0] = stbi__blinn_8x8(255 - out[0], m); + out[1] = stbi__blinn_8x8(255 - out[1], m); + out[2] = stbi__blinn_8x8(255 - out[2], m); + out += n; + } + } else { // YCbCr + alpha? Ignore the fourth channel for now + z->YCbCr_to_RGB_kernel(out, y, coutput[1], coutput[2], z->s->img_x, n); + } + } else + for (i = 0; i < z->s->img_x; ++i) { + out[0] = out[1] = out[2] = y[i]; + out[3] = 255; // not used if n==3 + out += n; + } + } else { + if (is_rgb) { + if (n == 1) + for (i = 0; i < z->s->img_x; ++i) + *out++ = stbi__compute_y(coutput[0][i], coutput[1][i], coutput[2][i]); + else { + for (i = 0; i < z->s->img_x; ++i, out += 2) { + out[0] = stbi__compute_y(coutput[0][i], coutput[1][i], coutput[2][i]); + out[1] = 255; + } + } + } else if (z->s->img_n == 4 && z->app14_color_transform == 0) { + for (i = 0; i < z->s->img_x; ++i) { + stbi_uc m = coutput[3][i]; + stbi_uc r = stbi__blinn_8x8(coutput[0][i], m); + stbi_uc g = stbi__blinn_8x8(coutput[1][i], m); + stbi_uc b = stbi__blinn_8x8(coutput[2][i], m); + out[0] = stbi__compute_y(r, g, b); + out[1] = 255; + out += n; + } + } else if (z->s->img_n == 4 && z->app14_color_transform == 2) { + for (i = 0; i < z->s->img_x; ++i) { + out[0] = stbi__blinn_8x8(255 - coutput[0][i], coutput[3][i]); + out[1] = 255; + out += n; + } + } else { + stbi_uc* y = coutput[0]; + if (n == 1) + for (i = 0; i < z->s->img_x; ++i) + out[i] = y[i]; + else + for (i = 0; i < z->s->img_x; ++i) { + *out++ = y[i]; + *out++ = 255; + } + } + } + } + stbi__cleanup_jpeg(z); + *out_x = z->s->img_x; + *out_y = z->s->img_y; + if (comp) + *comp = z->s->img_n >= 3 ? 3 : 1; // report original components, not output + return output; + } +} + +static void* stbi__jpeg_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri) +{ + unsigned char* result; + stbi__jpeg* j = (stbi__jpeg*)stbi__malloc(sizeof(stbi__jpeg)); + if (!j) + return stbi__errpuc("outofmem", "Out of memory"); + memset(j, 0, sizeof(stbi__jpeg)); + STBI_NOTUSED(ri); + j->s = s; + stbi__setup_jpeg(j); + result = load_jpeg_image(j, x, y, comp, req_comp); + STBI_FREE(j); + return result; +} + +static int stbi__jpeg_test(stbi__context* s) +{ + int r; + stbi__jpeg* j = (stbi__jpeg*)stbi__malloc(sizeof(stbi__jpeg)); + if (!j) + return stbi__err("outofmem", "Out of memory"); + memset(j, 0, sizeof(stbi__jpeg)); + j->s = s; + stbi__setup_jpeg(j); + r = stbi__decode_jpeg_header(j, STBI__SCAN_type); + stbi__rewind(s); + STBI_FREE(j); + return r; +} + +static int stbi__jpeg_info_raw(stbi__jpeg* j, int* x, int* y, int* comp) +{ + if (!stbi__decode_jpeg_header(j, STBI__SCAN_header)) { + stbi__rewind(j->s); + return 0; + } + if (x) + *x = j->s->img_x; + if (y) + *y = j->s->img_y; + if (comp) + *comp = j->s->img_n >= 3 ? 3 : 1; + return 1; +} + +static int stbi__jpeg_info(stbi__context* s, int* x, int* y, int* comp) +{ + int result; + stbi__jpeg* j = (stbi__jpeg*)(stbi__malloc(sizeof(stbi__jpeg))); + if (!j) + return stbi__err("outofmem", "Out of memory"); + memset(j, 0, sizeof(stbi__jpeg)); + j->s = s; + result = stbi__jpeg_info_raw(j, x, y, comp); + STBI_FREE(j); + return result; +} +# endif + +// public domain zlib decode v0.2 Sean Barrett 2006-11-18 +// simple implementation +// - all input must be provided in an upfront buffer +// - all output is written to a single output buffer (can malloc/realloc) +// performance +// - fast huffman + +# ifndef STBI_NO_ZLIB + +// fast-way is faster to check than jpeg huffman, but slow way is slower +# define STBI__ZFAST_BITS 9 // accelerate all cases in default tables +# define STBI__ZFAST_MASK ((1 << STBI__ZFAST_BITS) - 1) +# define STBI__ZNSYMS 288 // number of symbols in literal/length alphabet + +// zlib-style huffman encoding +// (jpegs packs from left, zlib from right, so can't share code) +typedef struct +{ + stbi__uint16 fast[1 << STBI__ZFAST_BITS]; + stbi__uint16 firstcode[16]; + int maxcode[17]; + stbi__uint16 firstsymbol[16]; + stbi_uc size[STBI__ZNSYMS]; + stbi__uint16 value[STBI__ZNSYMS]; +} stbi__zhuffman; + +stbi_inline static int stbi__bitreverse16(int n) +{ + n = ((n & 0xAAAA) >> 1) | ((n & 0x5555) << 1); + n = ((n & 0xCCCC) >> 2) | ((n & 0x3333) << 2); + n = ((n & 0xF0F0) >> 4) | ((n & 0x0F0F) << 4); + n = ((n & 0xFF00) >> 8) | ((n & 0x00FF) << 8); + return n; +} + +stbi_inline static int stbi__bit_reverse(int v, int bits) +{ + STBI_ASSERT(bits <= 16); + // to bit reverse n bits, reverse 16 and shift + // e.g. 11 bits, bit reverse and shift away 5 + return stbi__bitreverse16(v) >> (16 - bits); +} + +static int stbi__zbuild_huffman(stbi__zhuffman* z, stbi_uc const* sizelist, int num) +{ + int i, k = 0; + int code, next_code[16], sizes[17]; + + // DEFLATE spec for generating codes + memset(sizes, 0, sizeof(sizes)); + memset(z->fast, 0, sizeof(z->fast)); + for (i = 0; i < num; ++i) + ++sizes[sizelist[i]]; + sizes[0] = 0; + for (i = 1; i < 16; ++i) + if (sizes[i] > (1 << i)) + return stbi__err("bad sizes", "Corrupt PNG"); + code = 0; + for (i = 1; i < 16; ++i) { + next_code[i] = code; + z->firstcode[i] = (stbi__uint16)code; + z->firstsymbol[i] = (stbi__uint16)k; + code = (code + sizes[i]); + if (sizes[i]) + if (code - 1 >= (1 << i)) + return stbi__err("bad codelengths", "Corrupt PNG"); + z->maxcode[i] = code << (16 - i); // preshift for inner loop + code <<= 1; + k += sizes[i]; + } + z->maxcode[16] = 0x10000; // sentinel + for (i = 0; i < num; ++i) { + int s = sizelist[i]; + if (s) { + int c = next_code[s] - z->firstcode[s] + z->firstsymbol[s]; + stbi__uint16 fastv = (stbi__uint16)((s << 9) | i); + z->size[c] = (stbi_uc)s; + z->value[c] = (stbi__uint16)i; + if (s <= STBI__ZFAST_BITS) { + int j = stbi__bit_reverse(next_code[s], s); + while (j < (1 << STBI__ZFAST_BITS)) { + z->fast[j] = fastv; + j += (1 << s); + } + } + ++next_code[s]; + } + } + return 1; +} + +// zlib-from-memory implementation for PNG reading +// because PNG allows splitting the zlib stream arbitrarily, +// and it's annoying structurally to have PNG call ZLIB call PNG, +// we require PNG read all the IDATs and combine them into a single +// memory buffer + +typedef struct +{ + stbi_uc *zbuffer, *zbuffer_end; + int num_bits; + int hit_zeof_once; + stbi__uint32 code_buffer; + + char* zout; + char* zout_start; + char* zout_end; + int z_expandable; + + stbi__zhuffman z_length, z_distance; +} stbi__zbuf; + +stbi_inline static int stbi__zeof(stbi__zbuf* z) +{ + return (z->zbuffer >= z->zbuffer_end); +} + +stbi_inline static stbi_uc stbi__zget8(stbi__zbuf* z) +{ + return stbi__zeof(z) ? 0 : *z->zbuffer++; +} + +static void stbi__fill_bits(stbi__zbuf* z) +{ + do { + if (z->code_buffer >= (1U << z->num_bits)) { + z->zbuffer = z->zbuffer_end; /* treat this as EOF so we fail. */ + return; + } + z->code_buffer |= (unsigned int)stbi__zget8(z) << z->num_bits; + z->num_bits += 8; + } while (z->num_bits <= 24); +} + +stbi_inline static unsigned int stbi__zreceive(stbi__zbuf* z, int n) +{ + unsigned int k; + if (z->num_bits < n) + stbi__fill_bits(z); + k = z->code_buffer & ((1 << n) - 1); + z->code_buffer >>= n; + z->num_bits -= n; + return k; +} + +static int stbi__zhuffman_decode_slowpath(stbi__zbuf* a, stbi__zhuffman* z) +{ + int b, s, k; + // not resolved by fast table, so compute it the slow way + // use jpeg approach, which requires MSbits at top + k = stbi__bit_reverse(a->code_buffer, 16); + for (s = STBI__ZFAST_BITS + 1;; ++s) + if (k < z->maxcode[s]) + break; + if (s >= 16) + return -1; // invalid code! + // code size is s, so: + b = (k >> (16 - s)) - z->firstcode[s] + z->firstsymbol[s]; + if (b >= STBI__ZNSYMS) + return -1; // some data was corrupt somewhere! + if (z->size[b] != s) + return -1; // was originally an assert, but report failure instead. + a->code_buffer >>= s; + a->num_bits -= s; + return z->value[b]; +} + +stbi_inline static int stbi__zhuffman_decode(stbi__zbuf* a, stbi__zhuffman* z) +{ + int b, s; + if (a->num_bits < 16) { + if (stbi__zeof(a)) { + if (!a->hit_zeof_once) { + // This is the first time we hit eof, insert 16 extra padding btis + // to allow us to keep going; if we actually consume any of them + // though, that is invalid data. This is caught later. + a->hit_zeof_once = 1; + a->num_bits += 16; // add 16 implicit zero bits + } else { + // We already inserted our extra 16 padding bits and are again + // out, this stream is actually prematurely terminated. + return -1; + } + } else { + stbi__fill_bits(a); + } + } + b = z->fast[a->code_buffer & STBI__ZFAST_MASK]; + if (b) { + s = b >> 9; + a->code_buffer >>= s; + a->num_bits -= s; + return b & 511; + } + return stbi__zhuffman_decode_slowpath(a, z); +} + +static int stbi__zexpand(stbi__zbuf* z, char* zout, int n) // need to make room for n bytes +{ + char* q; + unsigned int cur, limit, old_limit; + z->zout = zout; + if (!z->z_expandable) + return stbi__err("output buffer limit", "Corrupt PNG"); + cur = (unsigned int)(z->zout - z->zout_start); + limit = old_limit = (unsigned)(z->zout_end - z->zout_start); + if (UINT_MAX - cur < (unsigned)n) + return stbi__err("outofmem", "Out of memory"); + while (cur + n > limit) { + if (limit > UINT_MAX / 2) + return stbi__err("outofmem", "Out of memory"); + limit *= 2; + } + q = (char*)STBI_REALLOC_SIZED(z->zout_start, old_limit, limit); + STBI_NOTUSED(old_limit); + if (q == NULL) + return stbi__err("outofmem", "Out of memory"); + z->zout_start = q; + z->zout = q + cur; + z->zout_end = q + limit; + return 1; +} + +static int const stbi__zlength_base[31] = { + 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, + 15, 17, 19, 23, 27, 31, 35, 43, 51, 59, + 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0 +}; + +static int const stbi__zlength_extra[31] = { 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, 0, 0 }; + +static int const stbi__zdist_base[32] = { 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193, + 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145, 8193, 12289, 16385, 24577, 0, 0 }; + +static int const stbi__zdist_extra[32] = { 0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13 }; + +static int stbi__parse_huffman_block(stbi__zbuf* a) +{ + char* zout = a->zout; + for (;;) { + int z = stbi__zhuffman_decode(a, &a->z_length); + if (z < 256) { + if (z < 0) + return stbi__err("bad huffman code", "Corrupt PNG"); // error in huffman codes + if (zout >= a->zout_end) { + if (!stbi__zexpand(a, zout, 1)) + return 0; + zout = a->zout; + } + *zout++ = (char)z; + } else { + stbi_uc* p; + int len, dist; + if (z == 256) { + a->zout = zout; + if (a->hit_zeof_once && a->num_bits < 16) { + // The first time we hit zeof, we inserted 16 extra zero bits into our bit + // buffer so the decoder can just do its speculative decoding. But if we + // actually consumed any of those bits (which is the case when num_bits < 16), + // the stream actually read past the end so it is malformed. + return stbi__err("unexpected end", "Corrupt PNG"); + } + return 1; + } + if (z >= 286) + return stbi__err("bad huffman code", "Corrupt PNG"); // per DEFLATE, length codes 286 and 287 must not appear in compressed data + z -= 257; + len = stbi__zlength_base[z]; + if (stbi__zlength_extra[z]) + len += stbi__zreceive(a, stbi__zlength_extra[z]); + z = stbi__zhuffman_decode(a, &a->z_distance); + if (z < 0 || z >= 30) + return stbi__err("bad huffman code", "Corrupt PNG"); // per DEFLATE, distance codes 30 and 31 must not appear in compressed data + dist = stbi__zdist_base[z]; + if (stbi__zdist_extra[z]) + dist += stbi__zreceive(a, stbi__zdist_extra[z]); + if (zout - a->zout_start < dist) + return stbi__err("bad dist", "Corrupt PNG"); + if (len > a->zout_end - zout) { + if (!stbi__zexpand(a, zout, len)) + return 0; + zout = a->zout; + } + p = (stbi_uc*)(zout - dist); + if (dist == 1) { // run of one byte; common in images. + stbi_uc v = *p; + if (len) { + do + *zout++ = v; + while (--len); + } + } else { + if (len) { + do + *zout++ = *p++; + while (--len); + } + } + } + } +} + +static int stbi__compute_huffman_codes(stbi__zbuf* a) +{ + static stbi_uc const length_dezigzag[19] = { 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15 }; + stbi__zhuffman z_codelength; + stbi_uc lencodes[286 + 32 + 137]; // padding for maximum single op + stbi_uc codelength_sizes[19]; + int i, n; + + int hlit = stbi__zreceive(a, 5) + 257; + int hdist = stbi__zreceive(a, 5) + 1; + int hclen = stbi__zreceive(a, 4) + 4; + int ntot = hlit + hdist; + + memset(codelength_sizes, 0, sizeof(codelength_sizes)); + for (i = 0; i < hclen; ++i) { + int s = stbi__zreceive(a, 3); + codelength_sizes[length_dezigzag[i]] = (stbi_uc)s; + } + if (!stbi__zbuild_huffman(&z_codelength, codelength_sizes, 19)) + return 0; + + n = 0; + while (n < ntot) { + int c = stbi__zhuffman_decode(a, &z_codelength); + if (c < 0 || c >= 19) + return stbi__err("bad codelengths", "Corrupt PNG"); + if (c < 16) + lencodes[n++] = (stbi_uc)c; + else { + stbi_uc fill = 0; + if (c == 16) { + c = stbi__zreceive(a, 2) + 3; + if (n == 0) + return stbi__err("bad codelengths", "Corrupt PNG"); + fill = lencodes[n - 1]; + } else if (c == 17) { + c = stbi__zreceive(a, 3) + 3; + } else if (c == 18) { + c = stbi__zreceive(a, 7) + 11; + } else { + return stbi__err("bad codelengths", "Corrupt PNG"); + } + if (ntot - n < c) + return stbi__err("bad codelengths", "Corrupt PNG"); + memset(lencodes + n, fill, c); + n += c; + } + } + if (n != ntot) + return stbi__err("bad codelengths", "Corrupt PNG"); + if (!stbi__zbuild_huffman(&a->z_length, lencodes, hlit)) + return 0; + if (!stbi__zbuild_huffman(&a->z_distance, lencodes + hlit, hdist)) + return 0; + return 1; +} + +static int stbi__parse_uncompressed_block(stbi__zbuf* a) +{ + stbi_uc header[4]; + int len, nlen, k; + if (a->num_bits & 7) + stbi__zreceive(a, a->num_bits & 7); // discard + // drain the bit-packed data into header + k = 0; + while (a->num_bits > 0) { + header[k++] = (stbi_uc)(a->code_buffer & 255); // suppress MSVC run-time check + a->code_buffer >>= 8; + a->num_bits -= 8; + } + if (a->num_bits < 0) + return stbi__err("zlib corrupt", "Corrupt PNG"); + // now fill header the normal way + while (k < 4) + header[k++] = stbi__zget8(a); + len = header[1] * 256 + header[0]; + nlen = header[3] * 256 + header[2]; + if (nlen != (len ^ 0xffff)) + return stbi__err("zlib corrupt", "Corrupt PNG"); + if (a->zbuffer + len > a->zbuffer_end) + return stbi__err("read past buffer", "Corrupt PNG"); + if (a->zout + len > a->zout_end) + if (!stbi__zexpand(a, a->zout, len)) + return 0; + memcpy(a->zout, a->zbuffer, len); + a->zbuffer += len; + a->zout += len; + return 1; +} + +static int stbi__parse_zlib_header(stbi__zbuf* a) +{ + int cmf = stbi__zget8(a); + int cm = cmf & 15; + /* int cinfo = cmf >> 4; */ + int flg = stbi__zget8(a); + if (stbi__zeof(a)) + return stbi__err("bad zlib header", "Corrupt PNG"); // zlib spec + if ((cmf * 256 + flg) % 31 != 0) + return stbi__err("bad zlib header", "Corrupt PNG"); // zlib spec + if (flg & 32) + return stbi__err("no preset dict", "Corrupt PNG"); // preset dictionary not allowed in png + if (cm != 8) + return stbi__err("bad compression", "Corrupt PNG"); // DEFLATE required for png + // window = 1 << (8 + cinfo)... but who cares, we fully buffer output + return 1; +} + +static stbi_uc const stbi__zdefault_length[STBI__ZNSYMS] = { + 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, + 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, + 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, + 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, + 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, + 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, + 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, + 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, + 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 8, 8, 8, 8, 8, 8, 8, 8 +}; +static stbi_uc const stbi__zdefault_distance[32] = { + 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5 +}; +/* +Init algorithm: +{ + int i; // use <= to match clearly with spec + for (i=0; i <= 143; ++i) stbi__zdefault_length[i] = 8; + for ( ; i <= 255; ++i) stbi__zdefault_length[i] = 9; + for ( ; i <= 279; ++i) stbi__zdefault_length[i] = 7; + for ( ; i <= 287; ++i) stbi__zdefault_length[i] = 8; + + for (i=0; i <= 31; ++i) stbi__zdefault_distance[i] = 5; +} +*/ + +static int stbi__parse_zlib(stbi__zbuf* a, int parse_header) +{ + int final, type; + if (parse_header) + if (!stbi__parse_zlib_header(a)) + return 0; + a->num_bits = 0; + a->code_buffer = 0; + a->hit_zeof_once = 0; + do { + final = stbi__zreceive(a, 1); + type = stbi__zreceive(a, 2); + if (type == 0) { + if (!stbi__parse_uncompressed_block(a)) + return 0; + } else if (type == 3) { + return 0; + } else { + if (type == 1) { + // use fixed code lengths + if (!stbi__zbuild_huffman(&a->z_length, stbi__zdefault_length, STBI__ZNSYMS)) + return 0; + if (!stbi__zbuild_huffman(&a->z_distance, stbi__zdefault_distance, 32)) + return 0; + } else { + if (!stbi__compute_huffman_codes(a)) + return 0; + } + if (!stbi__parse_huffman_block(a)) + return 0; + } + } while (!final); + return 1; +} + +static int stbi__do_zlib(stbi__zbuf* a, char* obuf, int olen, int exp, int parse_header) +{ + a->zout_start = obuf; + a->zout = obuf; + a->zout_end = obuf + olen; + a->z_expandable = exp; + + return stbi__parse_zlib(a, parse_header); +} + +STBIDEF char* stbi_zlib_decode_malloc_guesssize(char const* buffer, int len, int initial_size, int* outlen) +{ + stbi__zbuf a; + char* p = (char*)stbi__malloc(initial_size); + if (p == NULL) + return NULL; + a.zbuffer = (stbi_uc*)buffer; + a.zbuffer_end = (stbi_uc*)buffer + len; + if (stbi__do_zlib(&a, p, initial_size, 1, 1)) { + if (outlen) + *outlen = (int)(a.zout - a.zout_start); + return a.zout_start; + } else { + STBI_FREE(a.zout_start); + return NULL; + } +} + +STBIDEF char* stbi_zlib_decode_malloc(char const* buffer, int len, int* outlen) +{ + return stbi_zlib_decode_malloc_guesssize(buffer, len, 16384, outlen); +} + +STBIDEF char* stbi_zlib_decode_malloc_guesssize_headerflag(char const* buffer, int len, int initial_size, int* outlen, int parse_header) +{ + stbi__zbuf a; + char* p = (char*)stbi__malloc(initial_size); + if (p == NULL) + return NULL; + a.zbuffer = (stbi_uc*)buffer; + a.zbuffer_end = (stbi_uc*)buffer + len; + if (stbi__do_zlib(&a, p, initial_size, 1, parse_header)) { + if (outlen) + *outlen = (int)(a.zout - a.zout_start); + return a.zout_start; + } else { + STBI_FREE(a.zout_start); + return NULL; + } +} + +STBIDEF int stbi_zlib_decode_buffer(char* obuffer, int olen, char const* ibuffer, int ilen) +{ + stbi__zbuf a; + a.zbuffer = (stbi_uc*)ibuffer; + a.zbuffer_end = (stbi_uc*)ibuffer + ilen; + if (stbi__do_zlib(&a, obuffer, olen, 0, 1)) + return (int)(a.zout - a.zout_start); + else + return -1; +} + +STBIDEF char* stbi_zlib_decode_noheader_malloc(char const* buffer, int len, int* outlen) +{ + stbi__zbuf a; + char* p = (char*)stbi__malloc(16384); + if (p == NULL) + return NULL; + a.zbuffer = (stbi_uc*)buffer; + a.zbuffer_end = (stbi_uc*)buffer + len; + if (stbi__do_zlib(&a, p, 16384, 1, 0)) { + if (outlen) + *outlen = (int)(a.zout - a.zout_start); + return a.zout_start; + } else { + STBI_FREE(a.zout_start); + return NULL; + } +} + +STBIDEF int stbi_zlib_decode_noheader_buffer(char* obuffer, int olen, char const* ibuffer, int ilen) +{ + stbi__zbuf a; + a.zbuffer = (stbi_uc*)ibuffer; + a.zbuffer_end = (stbi_uc*)ibuffer + ilen; + if (stbi__do_zlib(&a, obuffer, olen, 0, 0)) + return (int)(a.zout - a.zout_start); + else + return -1; +} +# endif + +// public domain "baseline" PNG decoder v0.10 Sean Barrett 2006-11-18 +// simple implementation +// - only 8-bit samples +// - no CRC checking +// - allocates lots of intermediate memory +// - avoids problem of streaming data between subsystems +// - avoids explicit window management +// performance +// - uses stb_zlib, a PD zlib implementation with fast huffman decoding + +# ifndef STBI_NO_PNG +typedef struct +{ + stbi__uint32 length; + stbi__uint32 type; +} stbi__pngchunk; + +static stbi__pngchunk stbi__get_chunk_header(stbi__context* s) +{ + stbi__pngchunk c; + c.length = stbi__get32be(s); + c.type = stbi__get32be(s); + return c; +} + +static int stbi__check_png_header(stbi__context* s) +{ + static stbi_uc const png_sig[8] = { 137, 80, 78, 71, 13, 10, 26, 10 }; + int i; + for (i = 0; i < 8; ++i) + if (stbi__get8(s) != png_sig[i]) + return stbi__err("bad png sig", "Not a PNG"); + return 1; +} + +typedef struct +{ + stbi__context* s; + stbi_uc *idata, *expanded, *out; + int depth; +} stbi__png; + +enum { + STBI__F_none = 0, + STBI__F_sub = 1, + STBI__F_up = 2, + STBI__F_avg = 3, + STBI__F_paeth = 4, + // synthetic filter used for first scanline to avoid needing a dummy row of 0s + STBI__F_avg_first +}; + +static stbi_uc first_row_filter[5] = { + STBI__F_none, + STBI__F_sub, + STBI__F_none, + STBI__F_avg_first, + STBI__F_sub // Paeth with b=c=0 turns out to be equivalent to sub +}; + +static int stbi__paeth(int a, int b, int c) +{ + // This formulation looks very different from the reference in the PNG spec, but is + // actually equivalent and has favorable data dependencies and admits straightforward + // generation of branch-free code, which helps performance significantly. + int thresh = c * 3 - (a + b); + int lo = a < b ? a : b; + int hi = a < b ? b : a; + int t0 = (hi <= thresh) ? lo : c; + int t1 = (thresh <= lo) ? hi : t0; + return t1; +} + +static stbi_uc const stbi__depth_scale_table[9] = { 0, 0xff, 0x55, 0, 0x11, 0, 0, 0, 0x01 }; + +// adds an extra all-255 alpha channel +// dest == src is legal +// img_n must be 1 or 3 +static void stbi__create_png_alpha_expand8(stbi_uc* dest, stbi_uc* src, stbi__uint32 x, int img_n) +{ + int i; + // must process data backwards since we allow dest==src + if (img_n == 1) { + for (i = x - 1; i >= 0; --i) { + dest[i * 2 + 1] = 255; + dest[i * 2 + 0] = src[i]; + } + } else { + STBI_ASSERT(img_n == 3); + for (i = x - 1; i >= 0; --i) { + dest[i * 4 + 3] = 255; + dest[i * 4 + 2] = src[i * 3 + 2]; + dest[i * 4 + 1] = src[i * 3 + 1]; + dest[i * 4 + 0] = src[i * 3 + 0]; + } + } +} + +// create the png data from post-deflated data +static int stbi__create_png_image_raw(stbi__png* a, stbi_uc* raw, stbi__uint32 raw_len, int out_n, stbi__uint32 x, stbi__uint32 y, int depth, int color) +{ + int bytes = (depth == 16 ? 2 : 1); + stbi__context* s = a->s; + stbi__uint32 i, j, stride = x * out_n * bytes; + stbi__uint32 img_len, img_width_bytes; + stbi_uc* filter_buf; + int all_ok = 1; + int k; + int img_n = s->img_n; // copy it into a local for later + + int output_bytes = out_n * bytes; + int filter_bytes = img_n * bytes; + int width = x; + + STBI_ASSERT(out_n == s->img_n || out_n == s->img_n + 1); + a->out = (stbi_uc*)stbi__malloc_mad3(x, y, output_bytes, 0); // extra bytes to write off the end into + if (!a->out) + return stbi__err("outofmem", "Out of memory"); + + // note: error exits here don't need to clean up a->out individually, + // stbi__do_png always does on error. + if (!stbi__mad3sizes_valid(img_n, x, depth, 7)) + return stbi__err("too large", "Corrupt PNG"); + img_width_bytes = (((img_n * x * depth) + 7) >> 3); + if (!stbi__mad2sizes_valid(img_width_bytes, y, img_width_bytes)) + return stbi__err("too large", "Corrupt PNG"); + img_len = (img_width_bytes + 1) * y; + + // we used to check for exact match between raw_len and img_len on non-interlaced PNGs, + // but issue #276 reported a PNG in the wild that had extra data at the end (all zeros), + // so just check for raw_len < img_len always. + if (raw_len < img_len) + return stbi__err("not enough pixels", "Corrupt PNG"); + + // Allocate two scan lines worth of filter workspace buffer. + filter_buf = (stbi_uc*)stbi__malloc_mad2(img_width_bytes, 2, 0); + if (!filter_buf) + return stbi__err("outofmem", "Out of memory"); + + // Filtering for low-bit-depth images + if (depth < 8) { + filter_bytes = 1; + width = img_width_bytes; + } + + for (j = 0; j < y; ++j) { + // cur/prior filter buffers alternate + stbi_uc* cur = filter_buf + (j & 1) * img_width_bytes; + stbi_uc* prior = filter_buf + (~j & 1) * img_width_bytes; + stbi_uc* dest = a->out + stride * j; + int nk = width * filter_bytes; + int filter = *raw++; + + // check filter type + if (filter > 4) { + all_ok = stbi__err("invalid filter", "Corrupt PNG"); + break; + } + + // if first row, use special filter that doesn't sample previous row + if (j == 0) + filter = first_row_filter[filter]; + + // perform actual filtering + switch (filter) { + case STBI__F_none: + memcpy(cur, raw, nk); + break; + case STBI__F_sub: + memcpy(cur, raw, filter_bytes); + for (k = filter_bytes; k < nk; ++k) + cur[k] = STBI__BYTECAST(raw[k] + cur[k - filter_bytes]); + break; + case STBI__F_up: + for (k = 0; k < nk; ++k) + cur[k] = STBI__BYTECAST(raw[k] + prior[k]); + break; + case STBI__F_avg: + for (k = 0; k < filter_bytes; ++k) + cur[k] = STBI__BYTECAST(raw[k] + (prior[k] >> 1)); + for (k = filter_bytes; k < nk; ++k) + cur[k] = STBI__BYTECAST(raw[k] + ((prior[k] + cur[k - filter_bytes]) >> 1)); + break; + case STBI__F_paeth: + for (k = 0; k < filter_bytes; ++k) + cur[k] = STBI__BYTECAST(raw[k] + prior[k]); // prior[k] == stbi__paeth(0,prior[k],0) + for (k = filter_bytes; k < nk; ++k) + cur[k] = STBI__BYTECAST(raw[k] + stbi__paeth(cur[k - filter_bytes], prior[k], prior[k - filter_bytes])); + break; + case STBI__F_avg_first: + memcpy(cur, raw, filter_bytes); + for (k = filter_bytes; k < nk; ++k) + cur[k] = STBI__BYTECAST(raw[k] + (cur[k - filter_bytes] >> 1)); + break; + } + + raw += nk; + + // expand decoded bits in cur to dest, also adding an extra alpha channel if desired + if (depth < 8) { + stbi_uc scale = (color == 0) ? stbi__depth_scale_table[depth] : 1; // scale grayscale values to 0..255 range + stbi_uc* in = cur; + stbi_uc* out = dest; + stbi_uc inb = 0; + stbi__uint32 nsmp = x * img_n; + + // expand bits to bytes first + if (depth == 4) { + for (i = 0; i < nsmp; ++i) { + if ((i & 1) == 0) + inb = *in++; + *out++ = scale * (inb >> 4); + inb <<= 4; + } + } else if (depth == 2) { + for (i = 0; i < nsmp; ++i) { + if ((i & 3) == 0) + inb = *in++; + *out++ = scale * (inb >> 6); + inb <<= 2; + } + } else { + STBI_ASSERT(depth == 1); + for (i = 0; i < nsmp; ++i) { + if ((i & 7) == 0) + inb = *in++; + *out++ = scale * (inb >> 7); + inb <<= 1; + } + } + + // insert alpha=255 values if desired + if (img_n != out_n) + stbi__create_png_alpha_expand8(dest, dest, x, img_n); + } else if (depth == 8) { + if (img_n == out_n) + memcpy(dest, cur, x * img_n); + else + stbi__create_png_alpha_expand8(dest, cur, x, img_n); + } else if (depth == 16) { + // convert the image data from big-endian to platform-native + stbi__uint16* dest16 = (stbi__uint16*)dest; + stbi__uint32 nsmp = x * img_n; + + if (img_n == out_n) { + for (i = 0; i < nsmp; ++i, ++dest16, cur += 2) + *dest16 = (cur[0] << 8) | cur[1]; + } else { + STBI_ASSERT(img_n + 1 == out_n); + if (img_n == 1) { + for (i = 0; i < x; ++i, dest16 += 2, cur += 2) { + dest16[0] = (cur[0] << 8) | cur[1]; + dest16[1] = 0xffff; + } + } else { + STBI_ASSERT(img_n == 3); + for (i = 0; i < x; ++i, dest16 += 4, cur += 6) { + dest16[0] = (cur[0] << 8) | cur[1]; + dest16[1] = (cur[2] << 8) | cur[3]; + dest16[2] = (cur[4] << 8) | cur[5]; + dest16[3] = 0xffff; + } + } + } + } + } + + STBI_FREE(filter_buf); + if (!all_ok) + return 0; + + return 1; +} + +static int stbi__create_png_image(stbi__png* a, stbi_uc* image_data, stbi__uint32 image_data_len, int out_n, int depth, int color, int interlaced) +{ + int bytes = (depth == 16 ? 2 : 1); + int out_bytes = out_n * bytes; + stbi_uc* final; + int p; + if (!interlaced) + return stbi__create_png_image_raw(a, image_data, image_data_len, out_n, a->s->img_x, a->s->img_y, depth, color); + + // de-interlacing + final = (stbi_uc*)stbi__malloc_mad3(a->s->img_x, a->s->img_y, out_bytes, 0); + if (!final) + return stbi__err("outofmem", "Out of memory"); + for (p = 0; p < 7; ++p) { + int xorig[] = { 0, 4, 0, 2, 0, 1, 0 }; + int yorig[] = { 0, 0, 4, 0, 2, 0, 1 }; + int xspc[] = { 8, 8, 4, 4, 2, 2, 1 }; + int yspc[] = { 8, 8, 8, 4, 4, 2, 2 }; + int i, j, x, y; + // pass1_x[4] = 0, pass1_x[5] = 1, pass1_x[12] = 1 + x = (a->s->img_x - xorig[p] + xspc[p] - 1) / xspc[p]; + y = (a->s->img_y - yorig[p] + yspc[p] - 1) / yspc[p]; + if (x && y) { + stbi__uint32 img_len = ((((a->s->img_n * x * depth) + 7) >> 3) + 1) * y; + if (!stbi__create_png_image_raw(a, image_data, image_data_len, out_n, x, y, depth, color)) { + STBI_FREE(final); + return 0; + } + for (j = 0; j < y; ++j) { + for (i = 0; i < x; ++i) { + int out_y = j * yspc[p] + yorig[p]; + int out_x = i * xspc[p] + xorig[p]; + memcpy(final + out_y * a->s->img_x * out_bytes + out_x * out_bytes, + a->out + (j * x + i) * out_bytes, out_bytes); + } + } + STBI_FREE(a->out); + image_data += img_len; + image_data_len -= img_len; + } + } + a->out = final; + + return 1; +} + +static int stbi__compute_transparency(stbi__png* z, stbi_uc tc[3], int out_n) +{ + stbi__context* s = z->s; + stbi__uint32 i, pixel_count = s->img_x * s->img_y; + stbi_uc* p = z->out; + + // compute color-based transparency, assuming we've + // already got 255 as the alpha value in the output + STBI_ASSERT(out_n == 2 || out_n == 4); + + if (out_n == 2) { + for (i = 0; i < pixel_count; ++i) { + p[1] = (p[0] == tc[0] ? 0 : 255); + p += 2; + } + } else { + for (i = 0; i < pixel_count; ++i) { + if (p[0] == tc[0] && p[1] == tc[1] && p[2] == tc[2]) + p[3] = 0; + p += 4; + } + } + return 1; +} + +static int stbi__compute_transparency16(stbi__png* z, stbi__uint16 tc[3], int out_n) +{ + stbi__context* s = z->s; + stbi__uint32 i, pixel_count = s->img_x * s->img_y; + stbi__uint16* p = (stbi__uint16*)z->out; + + // compute color-based transparency, assuming we've + // already got 65535 as the alpha value in the output + STBI_ASSERT(out_n == 2 || out_n == 4); + + if (out_n == 2) { + for (i = 0; i < pixel_count; ++i) { + p[1] = (p[0] == tc[0] ? 0 : 65535); + p += 2; + } + } else { + for (i = 0; i < pixel_count; ++i) { + if (p[0] == tc[0] && p[1] == tc[1] && p[2] == tc[2]) + p[3] = 0; + p += 4; + } + } + return 1; +} + +static int stbi__expand_png_palette(stbi__png* a, stbi_uc* palette, int len, int pal_img_n) +{ + stbi__uint32 i, pixel_count = a->s->img_x * a->s->img_y; + stbi_uc *p, *temp_out, *orig = a->out; + + p = (stbi_uc*)stbi__malloc_mad2(pixel_count, pal_img_n, 0); + if (p == NULL) + return stbi__err("outofmem", "Out of memory"); + + // between here and free(out) below, exitting would leak + temp_out = p; + + if (pal_img_n == 3) { + for (i = 0; i < pixel_count; ++i) { + int n = orig[i] * 4; + p[0] = palette[n]; + p[1] = palette[n + 1]; + p[2] = palette[n + 2]; + p += 3; + } + } else { + for (i = 0; i < pixel_count; ++i) { + int n = orig[i] * 4; + p[0] = palette[n]; + p[1] = palette[n + 1]; + p[2] = palette[n + 2]; + p[3] = palette[n + 3]; + p += 4; + } + } + STBI_FREE(a->out); + a->out = temp_out; + + STBI_NOTUSED(len); + + return 1; +} + +static int stbi__unpremultiply_on_load_global = 0; +static int stbi__de_iphone_flag_global = 0; + +STBIDEF void stbi_set_unpremultiply_on_load(int flag_true_if_should_unpremultiply) +{ + stbi__unpremultiply_on_load_global = flag_true_if_should_unpremultiply; +} + +STBIDEF void stbi_convert_iphone_png_to_rgb(int flag_true_if_should_convert) +{ + stbi__de_iphone_flag_global = flag_true_if_should_convert; +} + +# ifndef STBI_THREAD_LOCAL +# define stbi__unpremultiply_on_load stbi__unpremultiply_on_load_global +# define stbi__de_iphone_flag stbi__de_iphone_flag_global +# else +static STBI_THREAD_LOCAL int stbi__unpremultiply_on_load_local, stbi__unpremultiply_on_load_set; +static STBI_THREAD_LOCAL int stbi__de_iphone_flag_local, stbi__de_iphone_flag_set; + +STBIDEF void stbi_set_unpremultiply_on_load_thread(int flag_true_if_should_unpremultiply) +{ + stbi__unpremultiply_on_load_local = flag_true_if_should_unpremultiply; + stbi__unpremultiply_on_load_set = 1; +} + +STBIDEF void stbi_convert_iphone_png_to_rgb_thread(int flag_true_if_should_convert) +{ + stbi__de_iphone_flag_local = flag_true_if_should_convert; + stbi__de_iphone_flag_set = 1; +} + +# define stbi__unpremultiply_on_load (stbi__unpremultiply_on_load_set \ + ? stbi__unpremultiply_on_load_local \ + : stbi__unpremultiply_on_load_global) +# define stbi__de_iphone_flag (stbi__de_iphone_flag_set \ + ? stbi__de_iphone_flag_local \ + : stbi__de_iphone_flag_global) +# endif // STBI_THREAD_LOCAL + +static void stbi__de_iphone(stbi__png* z) +{ + stbi__context* s = z->s; + stbi__uint32 i, pixel_count = s->img_x * s->img_y; + stbi_uc* p = z->out; + + if (s->img_out_n == 3) { // convert bgr to rgb + for (i = 0; i < pixel_count; ++i) { + stbi_uc t = p[0]; + p[0] = p[2]; + p[2] = t; + p += 3; + } + } else { + STBI_ASSERT(s->img_out_n == 4); + if (stbi__unpremultiply_on_load) { + // convert bgr to rgb and unpremultiply + for (i = 0; i < pixel_count; ++i) { + stbi_uc a = p[3]; + stbi_uc t = p[0]; + if (a) { + stbi_uc half = a / 2; + p[0] = (p[2] * 255 + half) / a; + p[1] = (p[1] * 255 + half) / a; + p[2] = (t * 255 + half) / a; + } else { + p[0] = p[2]; + p[2] = t; + } + p += 4; + } + } else { + // convert bgr to rgb + for (i = 0; i < pixel_count; ++i) { + stbi_uc t = p[0]; + p[0] = p[2]; + p[2] = t; + p += 4; + } + } + } +} + +# define STBI__PNG_TYPE(a, b, c, d) (((unsigned)(a) << 24) + ((unsigned)(b) << 16) + ((unsigned)(c) << 8) + (unsigned)(d)) + +static int stbi__parse_png_file(stbi__png* z, int scan, int req_comp) +{ + stbi_uc palette[1024], pal_img_n = 0; + stbi_uc has_trans = 0, tc[3] = { 0 }; + stbi__uint16 tc16[3]; + stbi__uint32 ioff = 0, idata_limit = 0, i, pal_len = 0; + int first = 1, k, interlace = 0, color = 0, is_iphone = 0; + stbi__context* s = z->s; + + z->expanded = NULL; + z->idata = NULL; + z->out = NULL; + + if (!stbi__check_png_header(s)) + return 0; + + if (scan == STBI__SCAN_type) + return 1; + + for (;;) { + stbi__pngchunk c = stbi__get_chunk_header(s); + switch (c.type) { + case STBI__PNG_TYPE('C', 'g', 'B', 'I'): + is_iphone = 1; + stbi__skip(s, c.length); + break; + case STBI__PNG_TYPE('I', 'H', 'D', 'R'): { + int comp, filter; + if (!first) + return stbi__err("multiple IHDR", "Corrupt PNG"); + first = 0; + if (c.length != 13) + return stbi__err("bad IHDR len", "Corrupt PNG"); + s->img_x = stbi__get32be(s); + s->img_y = stbi__get32be(s); + if (s->img_y > STBI_MAX_DIMENSIONS) + return stbi__err("too large", "Very large image (corrupt?)"); + if (s->img_x > STBI_MAX_DIMENSIONS) + return stbi__err("too large", "Very large image (corrupt?)"); + z->depth = stbi__get8(s); + if (z->depth != 1 && z->depth != 2 && z->depth != 4 && z->depth != 8 && z->depth != 16) + return stbi__err("1/2/4/8/16-bit only", "PNG not supported: 1/2/4/8/16-bit only"); + color = stbi__get8(s); + if (color > 6) + return stbi__err("bad ctype", "Corrupt PNG"); + if (color == 3 && z->depth == 16) + return stbi__err("bad ctype", "Corrupt PNG"); + if (color == 3) + pal_img_n = 3; + else if (color & 1) + return stbi__err("bad ctype", "Corrupt PNG"); + comp = stbi__get8(s); + if (comp) + return stbi__err("bad comp method", "Corrupt PNG"); + filter = stbi__get8(s); + if (filter) + return stbi__err("bad filter method", "Corrupt PNG"); + interlace = stbi__get8(s); + if (interlace > 1) + return stbi__err("bad interlace method", "Corrupt PNG"); + if (!s->img_x || !s->img_y) + return stbi__err("0-pixel image", "Corrupt PNG"); + if (!pal_img_n) { + s->img_n = (color & 2 ? 3 : 1) + (color & 4 ? 1 : 0); + if ((1 << 30) / s->img_x / s->img_n < s->img_y) + return stbi__err("too large", "Image too large to decode"); + } else { + // if paletted, then pal_n is our final components, and + // img_n is # components to decompress/filter. + s->img_n = 1; + if ((1 << 30) / s->img_x / 4 < s->img_y) + return stbi__err("too large", "Corrupt PNG"); + } + // even with SCAN_header, have to scan to see if we have a tRNS + break; + } + + case STBI__PNG_TYPE('P', 'L', 'T', 'E'): { + if (first) + return stbi__err("first not IHDR", "Corrupt PNG"); + if (c.length > 256 * 3) + return stbi__err("invalid PLTE", "Corrupt PNG"); + pal_len = c.length / 3; + if (pal_len * 3 != c.length) + return stbi__err("invalid PLTE", "Corrupt PNG"); + for (i = 0; i < pal_len; ++i) { + palette[i * 4 + 0] = stbi__get8(s); + palette[i * 4 + 1] = stbi__get8(s); + palette[i * 4 + 2] = stbi__get8(s); + palette[i * 4 + 3] = 255; + } + break; + } + + case STBI__PNG_TYPE('t', 'R', 'N', 'S'): { + if (first) + return stbi__err("first not IHDR", "Corrupt PNG"); + if (z->idata) + return stbi__err("tRNS after IDAT", "Corrupt PNG"); + if (pal_img_n) { + if (scan == STBI__SCAN_header) { + s->img_n = 4; + return 1; + } + if (pal_len == 0) + return stbi__err("tRNS before PLTE", "Corrupt PNG"); + if (c.length > pal_len) + return stbi__err("bad tRNS len", "Corrupt PNG"); + pal_img_n = 4; + for (i = 0; i < c.length; ++i) + palette[i * 4 + 3] = stbi__get8(s); + } else { + if (!(s->img_n & 1)) + return stbi__err("tRNS with alpha", "Corrupt PNG"); + if (c.length != (stbi__uint32)s->img_n * 2) + return stbi__err("bad tRNS len", "Corrupt PNG"); + has_trans = 1; + // non-paletted with tRNS = constant alpha. if header-scanning, we can stop now. + if (scan == STBI__SCAN_header) { + ++s->img_n; + return 1; + } + if (z->depth == 16) { + for (k = 0; k < s->img_n && k < 3; ++k) // extra loop test to suppress false GCC warning + tc16[k] = (stbi__uint16)stbi__get16be(s); // copy the values as-is + } else { + for (k = 0; k < s->img_n && k < 3; ++k) + tc[k] = (stbi_uc)(stbi__get16be(s) & 255) * stbi__depth_scale_table[z->depth]; // non 8-bit images will be larger + } + } + break; + } + + case STBI__PNG_TYPE('I', 'D', 'A', 'T'): { + if (first) + return stbi__err("first not IHDR", "Corrupt PNG"); + if (pal_img_n && !pal_len) + return stbi__err("no PLTE", "Corrupt PNG"); + if (scan == STBI__SCAN_header) { + // header scan definitely stops at first IDAT + if (pal_img_n) + s->img_n = pal_img_n; + return 1; + } + if (c.length > (1u << 30)) + return stbi__err("IDAT size limit", "IDAT section larger than 2^30 bytes"); + if ((int)(ioff + c.length) < (int)ioff) + return 0; + if (ioff + c.length > idata_limit) { + stbi__uint32 idata_limit_old = idata_limit; + stbi_uc* p; + if (idata_limit == 0) + idata_limit = c.length > 4096 ? c.length : 4096; + while (ioff + c.length > idata_limit) + idata_limit *= 2; + STBI_NOTUSED(idata_limit_old); + p = (stbi_uc*)STBI_REALLOC_SIZED(z->idata, idata_limit_old, idata_limit); + if (p == NULL) + return stbi__err("outofmem", "Out of memory"); + z->idata = p; + } + if (!stbi__getn(s, z->idata + ioff, c.length)) + return stbi__err("outofdata", "Corrupt PNG"); + ioff += c.length; + break; + } + + case STBI__PNG_TYPE('I', 'E', 'N', 'D'): { + stbi__uint32 raw_len, bpl; + if (first) + return stbi__err("first not IHDR", "Corrupt PNG"); + if (scan != STBI__SCAN_load) + return 1; + if (z->idata == NULL) + return stbi__err("no IDAT", "Corrupt PNG"); + // initial guess for decoded data size to avoid unnecessary reallocs + bpl = (s->img_x * z->depth + 7) / 8; // bytes per line, per component + raw_len = bpl * s->img_y * s->img_n /* pixels */ + s->img_y /* filter mode per row */; + z->expanded = (stbi_uc*)stbi_zlib_decode_malloc_guesssize_headerflag((char*)z->idata, ioff, raw_len, (int*)&raw_len, !is_iphone); + if (z->expanded == NULL) + return 0; // zlib should set error + STBI_FREE(z->idata); + z->idata = NULL; + if ((req_comp == s->img_n + 1 && req_comp != 3 && !pal_img_n) || has_trans) + s->img_out_n = s->img_n + 1; + else + s->img_out_n = s->img_n; + if (!stbi__create_png_image(z, z->expanded, raw_len, s->img_out_n, z->depth, color, interlace)) + return 0; + if (has_trans) { + if (z->depth == 16) { + if (!stbi__compute_transparency16(z, tc16, s->img_out_n)) + return 0; + } else { + if (!stbi__compute_transparency(z, tc, s->img_out_n)) + return 0; + } + } + if (is_iphone && stbi__de_iphone_flag && s->img_out_n > 2) + stbi__de_iphone(z); + if (pal_img_n) { + // pal_img_n == 3 or 4 + s->img_n = pal_img_n; // record the actual colors we had + s->img_out_n = pal_img_n; + if (req_comp >= 3) + s->img_out_n = req_comp; + if (!stbi__expand_png_palette(z, palette, pal_len, s->img_out_n)) + return 0; + } else if (has_trans) { + // non-paletted image with tRNS -> source image has (constant) alpha + ++s->img_n; + } + STBI_FREE(z->expanded); + z->expanded = NULL; + // end of PNG chunk, read and skip CRC + stbi__get32be(s); + return 1; + } + + default: + // if critical, fail + if (first) + return stbi__err("first not IHDR", "Corrupt PNG"); + if ((c.type & (1 << 29)) == 0) { +# ifndef STBI_NO_FAILURE_STRINGS + // not threadsafe + static char invalid_chunk[] = "XXXX PNG chunk not known"; + invalid_chunk[0] = STBI__BYTECAST(c.type >> 24); + invalid_chunk[1] = STBI__BYTECAST(c.type >> 16); + invalid_chunk[2] = STBI__BYTECAST(c.type >> 8); + invalid_chunk[3] = STBI__BYTECAST(c.type >> 0); +# endif + return stbi__err(invalid_chunk, "PNG not supported: unknown PNG chunk type"); + } + stbi__skip(s, c.length); + break; + } + // end of PNG chunk, read and skip CRC + stbi__get32be(s); + } +} + +static void* stbi__do_png(stbi__png* p, int* x, int* y, int* n, int req_comp, stbi__result_info* ri) +{ + void* result = NULL; + if (req_comp < 0 || req_comp > 4) + return stbi__errpuc("bad req_comp", "Internal error"); + if (stbi__parse_png_file(p, STBI__SCAN_load, req_comp)) { + if (p->depth <= 8) + ri->bits_per_channel = 8; + else if (p->depth == 16) + ri->bits_per_channel = 16; + else + return stbi__errpuc("bad bits_per_channel", "PNG not supported: unsupported color depth"); + result = p->out; + p->out = NULL; + if (req_comp && req_comp != p->s->img_out_n) { + if (ri->bits_per_channel == 8) + result = stbi__convert_format((unsigned char*)result, p->s->img_out_n, req_comp, p->s->img_x, p->s->img_y); + else + result = stbi__convert_format16((stbi__uint16*)result, p->s->img_out_n, req_comp, p->s->img_x, p->s->img_y); + p->s->img_out_n = req_comp; + if (result == NULL) + return result; + } + *x = p->s->img_x; + *y = p->s->img_y; + if (n) + *n = p->s->img_n; + } + STBI_FREE(p->out); + p->out = NULL; + STBI_FREE(p->expanded); + p->expanded = NULL; + STBI_FREE(p->idata); + p->idata = NULL; + + return result; +} + +static void* stbi__png_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri) +{ + stbi__png p; + p.s = s; + return stbi__do_png(&p, x, y, comp, req_comp, ri); +} + +static int stbi__png_test(stbi__context* s) +{ + int r; + r = stbi__check_png_header(s); + stbi__rewind(s); + return r; +} + +static int stbi__png_info_raw(stbi__png* p, int* x, int* y, int* comp) +{ + if (!stbi__parse_png_file(p, STBI__SCAN_header, 0)) { + stbi__rewind(p->s); + return 0; + } + if (x) + *x = p->s->img_x; + if (y) + *y = p->s->img_y; + if (comp) + *comp = p->s->img_n; + return 1; +} + +static int stbi__png_info(stbi__context* s, int* x, int* y, int* comp) +{ + stbi__png p; + p.s = s; + return stbi__png_info_raw(&p, x, y, comp); +} + +static int stbi__png_is16(stbi__context* s) +{ + stbi__png p; + p.s = s; + if (!stbi__png_info_raw(&p, NULL, NULL, NULL)) + return 0; + if (p.depth != 16) { + stbi__rewind(p.s); + return 0; + } + return 1; +} +# endif + +// Microsoft/Windows BMP image + +# ifndef STBI_NO_BMP +static int stbi__bmp_test_raw(stbi__context* s) +{ + int r; + int sz; + if (stbi__get8(s) != 'B') + return 0; + if (stbi__get8(s) != 'M') + return 0; + stbi__get32le(s); // discard filesize + stbi__get16le(s); // discard reserved + stbi__get16le(s); // discard reserved + stbi__get32le(s); // discard data offset + sz = stbi__get32le(s); + r = (sz == 12 || sz == 40 || sz == 56 || sz == 108 || sz == 124); + return r; +} + +static int stbi__bmp_test(stbi__context* s) +{ + int r = stbi__bmp_test_raw(s); + stbi__rewind(s); + return r; +} + +// returns 0..31 for the highest set bit +static int stbi__high_bit(unsigned int z) +{ + int n = 0; + if (z == 0) + return -1; + if (z >= 0x10000) { + n += 16; + z >>= 16; + } + if (z >= 0x00100) { + n += 8; + z >>= 8; + } + if (z >= 0x00010) { + n += 4; + z >>= 4; + } + if (z >= 0x00004) { + n += 2; + z >>= 2; + } + if (z >= 0x00002) { + n += 1; /* >>= 1;*/ + } + return n; +} + +static int stbi__bitcount(unsigned int a) +{ + a = (a & 0x55555555) + ((a >> 1) & 0x55555555); // max 2 + a = (a & 0x33333333) + ((a >> 2) & 0x33333333); // max 4 + a = (a + (a >> 4)) & 0x0f0f0f0f; // max 8 per 4, now 8 bits + a = (a + (a >> 8)); // max 16 per 8 bits + a = (a + (a >> 16)); // max 32 per 8 bits + return a & 0xff; +} + +// extract an arbitrarily-aligned N-bit value (N=bits) +// from v, and then make it 8-bits long and fractionally +// extend it to full full range. +static int stbi__shiftsigned(unsigned int v, int shift, int bits) +{ + static unsigned int mul_table[9] = { + 0, + 0xff /*0b11111111*/, + 0x55 /*0b01010101*/, + 0x49 /*0b01001001*/, + 0x11 /*0b00010001*/, + 0x21 /*0b00100001*/, + 0x41 /*0b01000001*/, + 0x81 /*0b10000001*/, + 0x01 /*0b00000001*/, + }; + static unsigned int shift_table[9] = { + 0, + 0, + 0, + 1, + 0, + 2, + 4, + 6, + 0, + }; + if (shift < 0) + v <<= -shift; + else + v >>= shift; + STBI_ASSERT(v < 256); + v >>= (8 - bits); + STBI_ASSERT(bits >= 0 && bits <= 8); + return (int)((unsigned)v * mul_table[bits]) >> shift_table[bits]; +} + +typedef struct +{ + int bpp, offset, hsz; + unsigned int mr, mg, mb, ma, all_a; + int extra_read; +} stbi__bmp_data; + +static int stbi__bmp_set_mask_defaults(stbi__bmp_data* info, int compress) +{ + // BI_BITFIELDS specifies masks explicitly, don't override + if (compress == 3) + return 1; + + if (compress == 0) { + if (info->bpp == 16) { + info->mr = 31u << 10; + info->mg = 31u << 5; + info->mb = 31u << 0; + } else if (info->bpp == 32) { + info->mr = 0xffu << 16; + info->mg = 0xffu << 8; + info->mb = 0xffu << 0; + info->ma = 0xffu << 24; + info->all_a = 0; // if all_a is 0 at end, then we loaded alpha channel but it was all 0 + } else { + // otherwise, use defaults, which is all-0 + info->mr = info->mg = info->mb = info->ma = 0; + } + return 1; + } + return 0; // error +} + +static void* stbi__bmp_parse_header(stbi__context* s, stbi__bmp_data* info) +{ + int hsz; + if (stbi__get8(s) != 'B' || stbi__get8(s) != 'M') + return stbi__errpuc("not BMP", "Corrupt BMP"); + stbi__get32le(s); // discard filesize + stbi__get16le(s); // discard reserved + stbi__get16le(s); // discard reserved + info->offset = stbi__get32le(s); + info->hsz = hsz = stbi__get32le(s); + info->mr = info->mg = info->mb = info->ma = 0; + info->extra_read = 14; + + if (info->offset < 0) + return stbi__errpuc("bad BMP", "bad BMP"); + + if (hsz != 12 && hsz != 40 && hsz != 56 && hsz != 108 && hsz != 124) + return stbi__errpuc("unknown BMP", "BMP type not supported: unknown"); + if (hsz == 12) { + s->img_x = stbi__get16le(s); + s->img_y = stbi__get16le(s); + } else { + s->img_x = stbi__get32le(s); + s->img_y = stbi__get32le(s); + } + if (stbi__get16le(s) != 1) + return stbi__errpuc("bad BMP", "bad BMP"); + info->bpp = stbi__get16le(s); + if (hsz != 12) { + int compress = stbi__get32le(s); + if (compress == 1 || compress == 2) + return stbi__errpuc("BMP RLE", "BMP type not supported: RLE"); + if (compress >= 4) + return stbi__errpuc("BMP JPEG/PNG", "BMP type not supported: unsupported compression"); // this includes PNG/JPEG modes + if (compress == 3 && info->bpp != 16 && info->bpp != 32) + return stbi__errpuc("bad BMP", "bad BMP"); // bitfields requires 16 or 32 bits/pixel + stbi__get32le(s); // discard sizeof + stbi__get32le(s); // discard hres + stbi__get32le(s); // discard vres + stbi__get32le(s); // discard colorsused + stbi__get32le(s); // discard max important + if (hsz == 40 || hsz == 56) { + if (hsz == 56) { + stbi__get32le(s); + stbi__get32le(s); + stbi__get32le(s); + stbi__get32le(s); + } + if (info->bpp == 16 || info->bpp == 32) { + if (compress == 0) { + stbi__bmp_set_mask_defaults(info, compress); + } else if (compress == 3) { + info->mr = stbi__get32le(s); + info->mg = stbi__get32le(s); + info->mb = stbi__get32le(s); + info->extra_read += 12; + // not documented, but generated by photoshop and handled by mspaint + if (info->mr == info->mg && info->mg == info->mb) { + // ?!?!? + return stbi__errpuc("bad BMP", "bad BMP"); + } + } else + return stbi__errpuc("bad BMP", "bad BMP"); + } + } else { + // V4/V5 header + int i; + if (hsz != 108 && hsz != 124) + return stbi__errpuc("bad BMP", "bad BMP"); + info->mr = stbi__get32le(s); + info->mg = stbi__get32le(s); + info->mb = stbi__get32le(s); + info->ma = stbi__get32le(s); + if (compress != 3) // override mr/mg/mb unless in BI_BITFIELDS mode, as per docs + stbi__bmp_set_mask_defaults(info, compress); + stbi__get32le(s); // discard color space + for (i = 0; i < 12; ++i) + stbi__get32le(s); // discard color space parameters + if (hsz == 124) { + stbi__get32le(s); // discard rendering intent + stbi__get32le(s); // discard offset of profile data + stbi__get32le(s); // discard size of profile data + stbi__get32le(s); // discard reserved + } + } + } + return (void*)1; +} + +static void* stbi__bmp_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri) +{ + stbi_uc* out; + unsigned int mr = 0, mg = 0, mb = 0, ma = 0, all_a; + stbi_uc pal[256][4]; + int psize = 0, i, j, width; + int flip_vertically, pad, target; + stbi__bmp_data info; + STBI_NOTUSED(ri); + + info.all_a = 255; + if (stbi__bmp_parse_header(s, &info) == NULL) + return NULL; // error code already set + + flip_vertically = ((int)s->img_y) > 0; + s->img_y = abs((int)s->img_y); + + if (s->img_y > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + if (s->img_x > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + + mr = info.mr; + mg = info.mg; + mb = info.mb; + ma = info.ma; + all_a = info.all_a; + + if (info.hsz == 12) { + if (info.bpp < 24) + psize = (info.offset - info.extra_read - 24) / 3; + } else { + if (info.bpp < 16) + psize = (info.offset - info.extra_read - info.hsz) >> 2; + } + if (psize == 0) { + // accept some number of extra bytes after the header, but if the offset points either to before + // the header ends or implies a large amount of extra data, reject the file as malformed + int bytes_read_so_far = s->callback_already_read + (int)(s->img_buffer - s->img_buffer_original); + int header_limit = 1024; // max we actually read is below 256 bytes currently. + int extra_data_limit = 256 * 4; // what ordinarily goes here is a palette; 256 entries*4 bytes is its max size. + if (bytes_read_so_far <= 0 || bytes_read_so_far > header_limit) { + return stbi__errpuc("bad header", "Corrupt BMP"); + } + // we established that bytes_read_so_far is positive and sensible. + // the first half of this test rejects offsets that are either too small positives, or + // negative, and guarantees that info.offset >= bytes_read_so_far > 0. this in turn + // ensures the number computed in the second half of the test can't overflow. + if (info.offset < bytes_read_so_far || info.offset - bytes_read_so_far > extra_data_limit) { + return stbi__errpuc("bad offset", "Corrupt BMP"); + } else { + stbi__skip(s, info.offset - bytes_read_so_far); + } + } + + if (info.bpp == 24 && ma == 0xff000000) + s->img_n = 3; + else + s->img_n = ma ? 4 : 3; + if (req_comp && req_comp >= 3) // we can directly decode 3 or 4 + target = req_comp; + else + target = s->img_n; // if they want monochrome, we'll post-convert + + // sanity-check size + if (!stbi__mad3sizes_valid(target, s->img_x, s->img_y, 0)) + return stbi__errpuc("too large", "Corrupt BMP"); + + out = (stbi_uc*)stbi__malloc_mad3(target, s->img_x, s->img_y, 0); + if (!out) + return stbi__errpuc("outofmem", "Out of memory"); + if (info.bpp < 16) { + int z = 0; + if (psize == 0 || psize > 256) { + STBI_FREE(out); + return stbi__errpuc("invalid", "Corrupt BMP"); + } + for (i = 0; i < psize; ++i) { + pal[i][2] = stbi__get8(s); + pal[i][1] = stbi__get8(s); + pal[i][0] = stbi__get8(s); + if (info.hsz != 12) + stbi__get8(s); + pal[i][3] = 255; + } + stbi__skip(s, info.offset - info.extra_read - info.hsz - psize * (info.hsz == 12 ? 3 : 4)); + if (info.bpp == 1) + width = (s->img_x + 7) >> 3; + else if (info.bpp == 4) + width = (s->img_x + 1) >> 1; + else if (info.bpp == 8) + width = s->img_x; + else { + STBI_FREE(out); + return stbi__errpuc("bad bpp", "Corrupt BMP"); + } + pad = (-width) & 3; + if (info.bpp == 1) { + for (j = 0; j < (int)s->img_y; ++j) { + int bit_offset = 7, v = stbi__get8(s); + for (i = 0; i < (int)s->img_x; ++i) { + int color = (v >> bit_offset) & 0x1; + out[z++] = pal[color][0]; + out[z++] = pal[color][1]; + out[z++] = pal[color][2]; + if (target == 4) + out[z++] = 255; + if (i + 1 == (int)s->img_x) + break; + if ((--bit_offset) < 0) { + bit_offset = 7; + v = stbi__get8(s); + } + } + stbi__skip(s, pad); + } + } else { + for (j = 0; j < (int)s->img_y; ++j) { + for (i = 0; i < (int)s->img_x; i += 2) { + int v = stbi__get8(s), v2 = 0; + if (info.bpp == 4) { + v2 = v & 15; + v >>= 4; + } + out[z++] = pal[v][0]; + out[z++] = pal[v][1]; + out[z++] = pal[v][2]; + if (target == 4) + out[z++] = 255; + if (i + 1 == (int)s->img_x) + break; + v = (info.bpp == 8) ? stbi__get8(s) : v2; + out[z++] = pal[v][0]; + out[z++] = pal[v][1]; + out[z++] = pal[v][2]; + if (target == 4) + out[z++] = 255; + } + stbi__skip(s, pad); + } + } + } else { + int rshift = 0, gshift = 0, bshift = 0, ashift = 0, rcount = 0, gcount = 0, bcount = 0, acount = 0; + int z = 0; + int easy = 0; + stbi__skip(s, info.offset - info.extra_read - info.hsz); + if (info.bpp == 24) + width = 3 * s->img_x; + else if (info.bpp == 16) + width = 2 * s->img_x; + else /* bpp = 32 and pad = 0 */ + width = 0; + pad = (-width) & 3; + if (info.bpp == 24) { + easy = 1; + } else if (info.bpp == 32) { + if (mb == 0xff && mg == 0xff00 && mr == 0x00ff0000 && ma == 0xff000000) + easy = 2; + } + if (!easy) { + if (!mr || !mg || !mb) { + STBI_FREE(out); + return stbi__errpuc("bad masks", "Corrupt BMP"); + } + // right shift amt to put high bit in position #7 + rshift = stbi__high_bit(mr) - 7; + rcount = stbi__bitcount(mr); + gshift = stbi__high_bit(mg) - 7; + gcount = stbi__bitcount(mg); + bshift = stbi__high_bit(mb) - 7; + bcount = stbi__bitcount(mb); + ashift = stbi__high_bit(ma) - 7; + acount = stbi__bitcount(ma); + if (rcount > 8 || gcount > 8 || bcount > 8 || acount > 8) { + STBI_FREE(out); + return stbi__errpuc("bad masks", "Corrupt BMP"); + } + } + for (j = 0; j < (int)s->img_y; ++j) { + if (easy) { + for (i = 0; i < (int)s->img_x; ++i) { + unsigned char a; + out[z + 2] = stbi__get8(s); + out[z + 1] = stbi__get8(s); + out[z + 0] = stbi__get8(s); + z += 3; + a = (easy == 2 ? stbi__get8(s) : 255); + all_a |= a; + if (target == 4) + out[z++] = a; + } + } else { + int bpp = info.bpp; + for (i = 0; i < (int)s->img_x; ++i) { + stbi__uint32 v = (bpp == 16 ? (stbi__uint32)stbi__get16le(s) : stbi__get32le(s)); + unsigned int a; + out[z++] = STBI__BYTECAST(stbi__shiftsigned(v & mr, rshift, rcount)); + out[z++] = STBI__BYTECAST(stbi__shiftsigned(v & mg, gshift, gcount)); + out[z++] = STBI__BYTECAST(stbi__shiftsigned(v & mb, bshift, bcount)); + a = (ma ? stbi__shiftsigned(v & ma, ashift, acount) : 255); + all_a |= a; + if (target == 4) + out[z++] = STBI__BYTECAST(a); + } + } + stbi__skip(s, pad); + } + } + + // if alpha channel is all 0s, replace with all 255s + if (target == 4 && all_a == 0) + for (i = 4 * s->img_x * s->img_y - 1; i >= 0; i -= 4) + out[i] = 255; + + if (flip_vertically) { + stbi_uc t; + for (j = 0; j < (int)s->img_y >> 1; ++j) { + stbi_uc* p1 = out + j * s->img_x * target; + stbi_uc* p2 = out + (s->img_y - 1 - j) * s->img_x * target; + for (i = 0; i < (int)s->img_x * target; ++i) { + t = p1[i]; + p1[i] = p2[i]; + p2[i] = t; + } + } + } + + if (req_comp && req_comp != target) { + out = stbi__convert_format(out, target, req_comp, s->img_x, s->img_y); + if (out == NULL) + return out; // stbi__convert_format frees input on failure + } + + *x = s->img_x; + *y = s->img_y; + if (comp) + *comp = s->img_n; + return out; +} +# endif + +// Targa Truevision - TGA +// by Jonathan Dummer +# ifndef STBI_NO_TGA +// returns STBI_rgb or whatever, 0 on error +static int stbi__tga_get_comp(int bits_per_pixel, int is_grey, int* is_rgb16) +{ + // only RGB or RGBA (incl. 16bit) or grey allowed + if (is_rgb16) + *is_rgb16 = 0; + switch (bits_per_pixel) { + case 8: + return STBI_grey; + case 16: + if (is_grey) + return STBI_grey_alpha; + // fallthrough + case 15: + if (is_rgb16) + *is_rgb16 = 1; + return STBI_rgb; + case 24: // fallthrough + case 32: + return bits_per_pixel / 8; + default: + return 0; + } +} + +static int stbi__tga_info(stbi__context* s, int* x, int* y, int* comp) +{ + int tga_w, tga_h, tga_comp, tga_image_type, tga_bits_per_pixel, tga_colormap_bpp; + int sz, tga_colormap_type; + stbi__get8(s); // discard Offset + tga_colormap_type = stbi__get8(s); // colormap type + if (tga_colormap_type > 1) { + stbi__rewind(s); + return 0; // only RGB or indexed allowed + } + tga_image_type = stbi__get8(s); // image type + if (tga_colormap_type == 1) { // colormapped (paletted) image + if (tga_image_type != 1 && tga_image_type != 9) { + stbi__rewind(s); + return 0; + } + stbi__skip(s, 4); // skip index of first colormap entry and number of entries + sz = stbi__get8(s); // check bits per palette color entry + if ((sz != 8) && (sz != 15) && (sz != 16) && (sz != 24) && (sz != 32)) { + stbi__rewind(s); + return 0; + } + stbi__skip(s, 4); // skip image x and y origin + tga_colormap_bpp = sz; + } else { // "normal" image w/o colormap - only RGB or grey allowed, +/- RLE + if ((tga_image_type != 2) && (tga_image_type != 3) && (tga_image_type != 10) && (tga_image_type != 11)) { + stbi__rewind(s); + return 0; // only RGB or grey allowed, +/- RLE + } + stbi__skip(s, 9); // skip colormap specification and image x/y origin + tga_colormap_bpp = 0; + } + tga_w = stbi__get16le(s); + if (tga_w < 1) { + stbi__rewind(s); + return 0; // test width + } + tga_h = stbi__get16le(s); + if (tga_h < 1) { + stbi__rewind(s); + return 0; // test height + } + tga_bits_per_pixel = stbi__get8(s); // bits per pixel + stbi__get8(s); // ignore alpha bits + if (tga_colormap_bpp != 0) { + if ((tga_bits_per_pixel != 8) && (tga_bits_per_pixel != 16)) { + // when using a colormap, tga_bits_per_pixel is the size of the indexes + // I don't think anything but 8 or 16bit indexes makes sense + stbi__rewind(s); + return 0; + } + tga_comp = stbi__tga_get_comp(tga_colormap_bpp, 0, NULL); + } else { + tga_comp = stbi__tga_get_comp(tga_bits_per_pixel, (tga_image_type == 3) || (tga_image_type == 11), NULL); + } + if (!tga_comp) { + stbi__rewind(s); + return 0; + } + if (x) + *x = tga_w; + if (y) + *y = tga_h; + if (comp) + *comp = tga_comp; + return 1; // seems to have passed everything +} + +static int stbi__tga_test(stbi__context* s) +{ + int res = 0; + int sz, tga_color_type; + stbi__get8(s); // discard Offset + tga_color_type = stbi__get8(s); // color type + if (tga_color_type > 1) + goto errorEnd; // only RGB or indexed allowed + sz = stbi__get8(s); // image type + if (tga_color_type == 1) { // colormapped (paletted) image + if (sz != 1 && sz != 9) + goto errorEnd; // colortype 1 demands image type 1 or 9 + stbi__skip(s, 4); // skip index of first colormap entry and number of entries + sz = stbi__get8(s); // check bits per palette color entry + if ((sz != 8) && (sz != 15) && (sz != 16) && (sz != 24) && (sz != 32)) + goto errorEnd; + stbi__skip(s, 4); // skip image x and y origin + } else { // "normal" image w/o colormap + if ((sz != 2) && (sz != 3) && (sz != 10) && (sz != 11)) + goto errorEnd; // only RGB or grey allowed, +/- RLE + stbi__skip(s, 9); // skip colormap specification and image x/y origin + } + if (stbi__get16le(s) < 1) + goto errorEnd; // test width + if (stbi__get16le(s) < 1) + goto errorEnd; // test height + sz = stbi__get8(s); // bits per pixel + if ((tga_color_type == 1) && (sz != 8) && (sz != 16)) + goto errorEnd; // for colormapped images, bpp is size of an index + if ((sz != 8) && (sz != 15) && (sz != 16) && (sz != 24) && (sz != 32)) + goto errorEnd; + + res = 1; // if we got this far, everything's good and we can return 1 instead of 0 + +errorEnd: + stbi__rewind(s); + return res; +} + +// read 16bit value and convert to 24bit RGB +static void stbi__tga_read_rgb16(stbi__context* s, stbi_uc* out) +{ + stbi__uint16 px = (stbi__uint16)stbi__get16le(s); + stbi__uint16 fiveBitMask = 31; + // we have 3 channels with 5bits each + int r = (px >> 10) & fiveBitMask; + int g = (px >> 5) & fiveBitMask; + int b = px & fiveBitMask; + // Note that this saves the data in RGB(A) order, so it doesn't need to be swapped later + out[0] = (stbi_uc)((r * 255) / 31); + out[1] = (stbi_uc)((g * 255) / 31); + out[2] = (stbi_uc)((b * 255) / 31); + + // some people claim that the most significant bit might be used for alpha + // (possibly if an alpha-bit is set in the "image descriptor byte") + // but that only made 16bit test images completely translucent.. + // so let's treat all 15 and 16bit TGAs as RGB with no alpha. +} + +static void* stbi__tga_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri) +{ + // read in the TGA header stuff + int tga_offset = stbi__get8(s); + int tga_indexed = stbi__get8(s); + int tga_image_type = stbi__get8(s); + int tga_is_RLE = 0; + int tga_palette_start = stbi__get16le(s); + int tga_palette_len = stbi__get16le(s); + int tga_palette_bits = stbi__get8(s); + int tga_x_origin = stbi__get16le(s); + int tga_y_origin = stbi__get16le(s); + int tga_width = stbi__get16le(s); + int tga_height = stbi__get16le(s); + int tga_bits_per_pixel = stbi__get8(s); + int tga_comp, tga_rgb16 = 0; + int tga_inverted = stbi__get8(s); + // int tga_alpha_bits = tga_inverted & 15; // the 4 lowest bits - unused (useless?) + // image data + unsigned char* tga_data; + unsigned char* tga_palette = NULL; + int i, j; + unsigned char raw_data[4] = { 0 }; + int RLE_count = 0; + int RLE_repeating = 0; + int read_next_pixel = 1; + STBI_NOTUSED(ri); + STBI_NOTUSED(tga_x_origin); // @TODO + STBI_NOTUSED(tga_y_origin); // @TODO + + if (tga_height > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + if (tga_width > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + + // do a tiny bit of precessing + if (tga_image_type >= 8) { + tga_image_type -= 8; + tga_is_RLE = 1; + } + tga_inverted = 1 - ((tga_inverted >> 5) & 1); + + // If I'm paletted, then I'll use the number of bits from the palette + if (tga_indexed) + tga_comp = stbi__tga_get_comp(tga_palette_bits, 0, &tga_rgb16); + else + tga_comp = stbi__tga_get_comp(tga_bits_per_pixel, (tga_image_type == 3), &tga_rgb16); + + if (!tga_comp) // shouldn't really happen, stbi__tga_test() should have ensured basic consistency + return stbi__errpuc("bad format", "Can't find out TGA pixelformat"); + + // tga info + *x = tga_width; + *y = tga_height; + if (comp) + *comp = tga_comp; + + if (!stbi__mad3sizes_valid(tga_width, tga_height, tga_comp, 0)) + return stbi__errpuc("too large", "Corrupt TGA"); + + tga_data = (unsigned char*)stbi__malloc_mad3(tga_width, tga_height, tga_comp, 0); + if (!tga_data) + return stbi__errpuc("outofmem", "Out of memory"); + + // skip to the data's starting position (offset usually = 0) + stbi__skip(s, tga_offset); + + if (!tga_indexed && !tga_is_RLE && !tga_rgb16) { + for (i = 0; i < tga_height; ++i) { + int row = tga_inverted ? tga_height - i - 1 : i; + stbi_uc* tga_row = tga_data + row * tga_width * tga_comp; + stbi__getn(s, tga_row, tga_width * tga_comp); + } + } else { + // do I need to load a palette? + if (tga_indexed) { + if (tga_palette_len == 0) { /* you have to have at least one entry! */ + STBI_FREE(tga_data); + return stbi__errpuc("bad palette", "Corrupt TGA"); + } + + // any data to skip? (offset usually = 0) + stbi__skip(s, tga_palette_start); + // load the palette + tga_palette = (unsigned char*)stbi__malloc_mad2(tga_palette_len, tga_comp, 0); + if (!tga_palette) { + STBI_FREE(tga_data); + return stbi__errpuc("outofmem", "Out of memory"); + } + if (tga_rgb16) { + stbi_uc* pal_entry = tga_palette; + STBI_ASSERT(tga_comp == STBI_rgb); + for (i = 0; i < tga_palette_len; ++i) { + stbi__tga_read_rgb16(s, pal_entry); + pal_entry += tga_comp; + } + } else if (!stbi__getn(s, tga_palette, tga_palette_len * tga_comp)) { + STBI_FREE(tga_data); + STBI_FREE(tga_palette); + return stbi__errpuc("bad palette", "Corrupt TGA"); + } + } + // load the data + for (i = 0; i < tga_width * tga_height; ++i) { + // if I'm in RLE mode, do I need to get a RLE stbi__pngchunk? + if (tga_is_RLE) { + if (RLE_count == 0) { + // yep, get the next byte as a RLE command + int RLE_cmd = stbi__get8(s); + RLE_count = 1 + (RLE_cmd & 127); + RLE_repeating = RLE_cmd >> 7; + read_next_pixel = 1; + } else if (!RLE_repeating) { + read_next_pixel = 1; + } + } else { + read_next_pixel = 1; + } + // OK, if I need to read a pixel, do it now + if (read_next_pixel) { + // load however much data we did have + if (tga_indexed) { + // read in index, then perform the lookup + int pal_idx = (tga_bits_per_pixel == 8) ? stbi__get8(s) : stbi__get16le(s); + if (pal_idx >= tga_palette_len) { + // invalid index + pal_idx = 0; + } + pal_idx *= tga_comp; + for (j = 0; j < tga_comp; ++j) { + raw_data[j] = tga_palette[pal_idx + j]; + } + } else if (tga_rgb16) { + STBI_ASSERT(tga_comp == STBI_rgb); + stbi__tga_read_rgb16(s, raw_data); + } else { + // read in the data raw + for (j = 0; j < tga_comp; ++j) { + raw_data[j] = stbi__get8(s); + } + } + // clear the reading flag for the next pixel + read_next_pixel = 0; + } // end of reading a pixel + + // copy data + for (j = 0; j < tga_comp; ++j) + tga_data[i * tga_comp + j] = raw_data[j]; + + // in case we're in RLE mode, keep counting down + --RLE_count; + } + // do I need to invert the image? + if (tga_inverted) { + for (j = 0; j * 2 < tga_height; ++j) { + int index1 = j * tga_width * tga_comp; + int index2 = (tga_height - 1 - j) * tga_width * tga_comp; + for (i = tga_width * tga_comp; i > 0; --i) { + unsigned char temp = tga_data[index1]; + tga_data[index1] = tga_data[index2]; + tga_data[index2] = temp; + ++index1; + ++index2; + } + } + } + // clear my palette, if I had one + if (tga_palette != NULL) { + STBI_FREE(tga_palette); + } + } + + // swap RGB - if the source data was RGB16, it already is in the right order + if (tga_comp >= 3 && !tga_rgb16) { + unsigned char* tga_pixel = tga_data; + for (i = 0; i < tga_width * tga_height; ++i) { + unsigned char temp = tga_pixel[0]; + tga_pixel[0] = tga_pixel[2]; + tga_pixel[2] = temp; + tga_pixel += tga_comp; + } + } + + // convert to target component count + if (req_comp && req_comp != tga_comp) + tga_data = stbi__convert_format(tga_data, tga_comp, req_comp, tga_width, tga_height); + + // the things I do to get rid of an error message, and yet keep + // Microsoft's C compilers happy... [8^( + tga_palette_start = tga_palette_len = tga_palette_bits = tga_x_origin = tga_y_origin = 0; + STBI_NOTUSED(tga_palette_start); + // OK, done + return tga_data; +} +# endif + +// ************************************************************************************************* +// Photoshop PSD loader -- PD by Thatcher Ulrich, integration by Nicolas Schulz, tweaked by STB + +# ifndef STBI_NO_PSD +static int stbi__psd_test(stbi__context* s) +{ + int r = (stbi__get32be(s) == 0x38425053); + stbi__rewind(s); + return r; +} + +static int stbi__psd_decode_rle(stbi__context* s, stbi_uc* p, int pixelCount) +{ + int count, nleft, len; + + count = 0; + while ((nleft = pixelCount - count) > 0) { + len = stbi__get8(s); + if (len == 128) { + // No-op. + } else if (len < 128) { + // Copy next len+1 bytes literally. + len++; + if (len > nleft) + return 0; // corrupt data + count += len; + while (len) { + *p = stbi__get8(s); + p += 4; + len--; + } + } else if (len > 128) { + stbi_uc val; + // Next -len+1 bytes in the dest are replicated from next source byte. + // (Interpret len as a negative 8-bit int.) + len = 257 - len; + if (len > nleft) + return 0; // corrupt data + val = stbi__get8(s); + count += len; + while (len) { + *p = val; + p += 4; + len--; + } + } + } + + return 1; +} + +static void* stbi__psd_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri, int bpc) +{ + int pixelCount; + int channelCount, compression; + int channel, i; + int bitdepth; + int w, h; + stbi_uc* out; + STBI_NOTUSED(ri); + + // Check identifier + if (stbi__get32be(s) != 0x38425053) // "8BPS" + return stbi__errpuc("not PSD", "Corrupt PSD image"); + + // Check file type version. + if (stbi__get16be(s) != 1) + return stbi__errpuc("wrong version", "Unsupported version of PSD image"); + + // Skip 6 reserved bytes. + stbi__skip(s, 6); + + // Read the number of channels (R, G, B, A, etc). + channelCount = stbi__get16be(s); + if (channelCount < 0 || channelCount > 16) + return stbi__errpuc("wrong channel count", "Unsupported number of channels in PSD image"); + + // Read the rows and columns of the image. + h = stbi__get32be(s); + w = stbi__get32be(s); + + if (h > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + if (w > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + + // Make sure the depth is 8 bits. + bitdepth = stbi__get16be(s); + if (bitdepth != 8 && bitdepth != 16) + return stbi__errpuc("unsupported bit depth", "PSD bit depth is not 8 or 16 bit"); + + // Make sure the color mode is RGB. + // Valid options are: + // 0: Bitmap + // 1: Grayscale + // 2: Indexed color + // 3: RGB color + // 4: CMYK color + // 7: Multichannel + // 8: Duotone + // 9: Lab color + if (stbi__get16be(s) != 3) + return stbi__errpuc("wrong color format", "PSD is not in RGB color format"); + + // Skip the Mode Data. (It's the palette for indexed color; other info for other modes.) + stbi__skip(s, stbi__get32be(s)); + + // Skip the image resources. (resolution, pen tool paths, etc) + stbi__skip(s, stbi__get32be(s)); + + // Skip the reserved data. + stbi__skip(s, stbi__get32be(s)); + + // Find out if the data is compressed. + // Known values: + // 0: no compression + // 1: RLE compressed + compression = stbi__get16be(s); + if (compression > 1) + return stbi__errpuc("bad compression", "PSD has an unknown compression format"); + + // Check size + if (!stbi__mad3sizes_valid(4, w, h, 0)) + return stbi__errpuc("too large", "Corrupt PSD"); + + // Create the destination image. + + if (!compression && bitdepth == 16 && bpc == 16) { + out = (stbi_uc*)stbi__malloc_mad3(8, w, h, 0); + ri->bits_per_channel = 16; + } else + out = (stbi_uc*)stbi__malloc(4 * w * h); + + if (!out) + return stbi__errpuc("outofmem", "Out of memory"); + pixelCount = w * h; + + // Initialize the data to zero. + // memset( out, 0, pixelCount * 4 ); + + // Finally, the image data. + if (compression) { + // RLE as used by .PSD and .TIFF + // Loop until you get the number of unpacked bytes you are expecting: + // Read the next source byte into n. + // If n is between 0 and 127 inclusive, copy the next n+1 bytes literally. + // Else if n is between -127 and -1 inclusive, copy the next byte -n+1 times. + // Else if n is 128, noop. + // Endloop + + // The RLE-compressed data is preceded by a 2-byte data count for each row in the data, + // which we're going to just skip. + stbi__skip(s, h * channelCount * 2); + + // Read the RLE data by channel. + for (channel = 0; channel < 4; channel++) { + stbi_uc* p; + + p = out + channel; + if (channel >= channelCount) { + // Fill this channel with default data. + for (i = 0; i < pixelCount; i++, p += 4) + *p = (channel == 3 ? 255 : 0); + } else { + // Read the RLE data. + if (!stbi__psd_decode_rle(s, p, pixelCount)) { + STBI_FREE(out); + return stbi__errpuc("corrupt", "bad RLE data"); + } + } + } + + } else { + // We're at the raw image data. It's each channel in order (Red, Green, Blue, Alpha, ...) + // where each channel consists of an 8-bit (or 16-bit) value for each pixel in the image. + + // Read the data by channel. + for (channel = 0; channel < 4; channel++) { + if (channel >= channelCount) { + // Fill this channel with default data. + if (bitdepth == 16 && bpc == 16) { + stbi__uint16* q = ((stbi__uint16*)out) + channel; + stbi__uint16 val = channel == 3 ? 65535 : 0; + for (i = 0; i < pixelCount; i++, q += 4) + *q = val; + } else { + stbi_uc* p = out + channel; + stbi_uc val = channel == 3 ? 255 : 0; + for (i = 0; i < pixelCount; i++, p += 4) + *p = val; + } + } else { + if (ri->bits_per_channel == 16) { // output bpc + stbi__uint16* q = ((stbi__uint16*)out) + channel; + for (i = 0; i < pixelCount; i++, q += 4) + *q = (stbi__uint16)stbi__get16be(s); + } else { + stbi_uc* p = out + channel; + if (bitdepth == 16) { // input bpc + for (i = 0; i < pixelCount; i++, p += 4) + *p = (stbi_uc)(stbi__get16be(s) >> 8); + } else { + for (i = 0; i < pixelCount; i++, p += 4) + *p = stbi__get8(s); + } + } + } + } + } + + // remove weird white matte from PSD + if (channelCount >= 4) { + if (ri->bits_per_channel == 16) { + for (i = 0; i < w * h; ++i) { + stbi__uint16* pixel = (stbi__uint16*)out + 4 * i; + if (pixel[3] != 0 && pixel[3] != 65535) { + float a = pixel[3] / 65535.0f; + float ra = 1.0f / a; + float inv_a = 65535.0f * (1 - ra); + pixel[0] = (stbi__uint16)(pixel[0] * ra + inv_a); + pixel[1] = (stbi__uint16)(pixel[1] * ra + inv_a); + pixel[2] = (stbi__uint16)(pixel[2] * ra + inv_a); + } + } + } else { + for (i = 0; i < w * h; ++i) { + unsigned char* pixel = out + 4 * i; + if (pixel[3] != 0 && pixel[3] != 255) { + float a = pixel[3] / 255.0f; + float ra = 1.0f / a; + float inv_a = 255.0f * (1 - ra); + pixel[0] = (unsigned char)(pixel[0] * ra + inv_a); + pixel[1] = (unsigned char)(pixel[1] * ra + inv_a); + pixel[2] = (unsigned char)(pixel[2] * ra + inv_a); + } + } + } + } + + // convert to desired output format + if (req_comp && req_comp != 4) { + if (ri->bits_per_channel == 16) + out = (stbi_uc*)stbi__convert_format16((stbi__uint16*)out, 4, req_comp, w, h); + else + out = stbi__convert_format(out, 4, req_comp, w, h); + if (out == NULL) + return out; // stbi__convert_format frees input on failure + } + + if (comp) + *comp = 4; + *y = h; + *x = w; + + return out; +} +# endif + +// ************************************************************************************************* +// Softimage PIC loader +// by Tom Seddon +// +// See http://softimage.wiki.softimage.com/index.php/INFO:_PIC_file_format +// See http://ozviz.wasp.uwa.edu.au/~pbourke/dataformats/softimagepic/ + +# ifndef STBI_NO_PIC +static int stbi__pic_is4(stbi__context* s, char const* str) +{ + int i; + for (i = 0; i < 4; ++i) + if (stbi__get8(s) != (stbi_uc)str[i]) + return 0; + + return 1; +} + +static int stbi__pic_test_core(stbi__context* s) +{ + int i; + + if (!stbi__pic_is4(s, "\x53\x80\xF6\x34")) + return 0; + + for (i = 0; i < 84; ++i) + stbi__get8(s); + + if (!stbi__pic_is4(s, "PICT")) + return 0; + + return 1; +} + +typedef struct +{ + stbi_uc size, type, channel; +} stbi__pic_packet; + +static stbi_uc* stbi__readval(stbi__context* s, int channel, stbi_uc* dest) +{ + int mask = 0x80, i; + + for (i = 0; i < 4; ++i, mask >>= 1) { + if (channel & mask) { + if (stbi__at_eof(s)) + return stbi__errpuc("bad file", "PIC file too short"); + dest[i] = stbi__get8(s); + } + } + + return dest; +} + +static void stbi__copyval(int channel, stbi_uc* dest, stbi_uc const* src) +{ + int mask = 0x80, i; + + for (i = 0; i < 4; ++i, mask >>= 1) + if (channel & mask) + dest[i] = src[i]; +} + +static stbi_uc* stbi__pic_load_core(stbi__context* s, int width, int height, int* comp, stbi_uc* result) +{ + int act_comp = 0, num_packets = 0, y, chained; + stbi__pic_packet packets[10]; + + // this will (should...) cater for even some bizarre stuff like having data + // for the same channel in multiple packets. + do { + stbi__pic_packet* packet; + + if (num_packets == sizeof(packets) / sizeof(packets[0])) + return stbi__errpuc("bad format", "too many packets"); + + packet = &packets[num_packets++]; + + chained = stbi__get8(s); + packet->size = stbi__get8(s); + packet->type = stbi__get8(s); + packet->channel = stbi__get8(s); + + act_comp |= packet->channel; + + if (stbi__at_eof(s)) + return stbi__errpuc("bad file", "file too short (reading packets)"); + if (packet->size != 8) + return stbi__errpuc("bad format", "packet isn't 8bpp"); + } while (chained); + + *comp = (act_comp & 0x10 ? 4 : 3); // has alpha channel? + + for (y = 0; y < height; ++y) { + int packet_idx; + + for (packet_idx = 0; packet_idx < num_packets; ++packet_idx) { + stbi__pic_packet* packet = &packets[packet_idx]; + stbi_uc* dest = result + y * width * 4; + + switch (packet->type) { + default: + return stbi__errpuc("bad format", "packet has bad compression type"); + + case 0: { // uncompressed + int x; + + for (x = 0; x < width; ++x, dest += 4) + if (!stbi__readval(s, packet->channel, dest)) + return 0; + break; + } + + case 1: // Pure RLE + { + int left = width, i; + + while (left > 0) { + stbi_uc count, value[4]; + + count = stbi__get8(s); + if (stbi__at_eof(s)) + return stbi__errpuc("bad file", "file too short (pure read count)"); + + if (count > left) + count = (stbi_uc)left; + + if (!stbi__readval(s, packet->channel, value)) + return 0; + + for (i = 0; i < count; ++i, dest += 4) + stbi__copyval(packet->channel, dest, value); + left -= count; + } + } break; + + case 2: { // Mixed RLE + int left = width; + while (left > 0) { + int count = stbi__get8(s), i; + if (stbi__at_eof(s)) + return stbi__errpuc("bad file", "file too short (mixed read count)"); + + if (count >= 128) { // Repeated + stbi_uc value[4]; + + if (count == 128) + count = stbi__get16be(s); + else + count -= 127; + if (count > left) + return stbi__errpuc("bad file", "scanline overrun"); + + if (!stbi__readval(s, packet->channel, value)) + return 0; + + for (i = 0; i < count; ++i, dest += 4) + stbi__copyval(packet->channel, dest, value); + } else { // Raw + ++count; + if (count > left) + return stbi__errpuc("bad file", "scanline overrun"); + + for (i = 0; i < count; ++i, dest += 4) + if (!stbi__readval(s, packet->channel, dest)) + return 0; + } + left -= count; + } + break; + } + } + } + } + + return result; +} + +static void* stbi__pic_load(stbi__context* s, int* px, int* py, int* comp, int req_comp, stbi__result_info* ri) +{ + stbi_uc* result; + int i, x, y, internal_comp; + STBI_NOTUSED(ri); + + if (!comp) + comp = &internal_comp; + + for (i = 0; i < 92; ++i) + stbi__get8(s); + + x = stbi__get16be(s); + y = stbi__get16be(s); + + if (y > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + if (x > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + + if (stbi__at_eof(s)) + return stbi__errpuc("bad file", "file too short (pic header)"); + if (!stbi__mad3sizes_valid(x, y, 4, 0)) + return stbi__errpuc("too large", "PIC image too large to decode"); + + stbi__get32be(s); // skip `ratio' + stbi__get16be(s); // skip `fields' + stbi__get16be(s); // skip `pad' + + // intermediate buffer is RGBA + result = (stbi_uc*)stbi__malloc_mad3(x, y, 4, 0); + if (!result) + return stbi__errpuc("outofmem", "Out of memory"); + memset(result, 0xff, x * y * 4); + + if (!stbi__pic_load_core(s, x, y, comp, result)) { + STBI_FREE(result); + result = 0; + } + *px = x; + *py = y; + if (req_comp == 0) + req_comp = *comp; + result = stbi__convert_format(result, 4, req_comp, x, y); + + return result; +} + +static int stbi__pic_test(stbi__context* s) +{ + int r = stbi__pic_test_core(s); + stbi__rewind(s); + return r; +} +# endif + +// ************************************************************************************************* +// GIF loader -- public domain by Jean-Marc Lienher -- simplified/shrunk by stb + +# ifndef STBI_NO_GIF +typedef struct +{ + stbi__int16 prefix; + stbi_uc first; + stbi_uc suffix; +} stbi__gif_lzw; + +typedef struct +{ + int w, h; + stbi_uc* out; // output buffer (always 4 components) + stbi_uc* background; // The current "background" as far as a gif is concerned + stbi_uc* history; + int flags, bgindex, ratio, transparent, eflags; + stbi_uc pal[256][4]; + stbi_uc lpal[256][4]; + stbi__gif_lzw codes[8192]; + stbi_uc* color_table; + int parse, step; + int lflags; + int start_x, start_y; + int max_x, max_y; + int cur_x, cur_y; + int line_size; + int delay; +} stbi__gif; + +static int stbi__gif_test_raw(stbi__context* s) +{ + int sz; + if (stbi__get8(s) != 'G' || stbi__get8(s) != 'I' || stbi__get8(s) != 'F' || stbi__get8(s) != '8') + return 0; + sz = stbi__get8(s); + if (sz != '9' && sz != '7') + return 0; + if (stbi__get8(s) != 'a') + return 0; + return 1; +} + +static int stbi__gif_test(stbi__context* s) +{ + int r = stbi__gif_test_raw(s); + stbi__rewind(s); + return r; +} + +static void stbi__gif_parse_colortable(stbi__context* s, stbi_uc pal[256][4], int num_entries, int transp) +{ + int i; + for (i = 0; i < num_entries; ++i) { + pal[i][2] = stbi__get8(s); + pal[i][1] = stbi__get8(s); + pal[i][0] = stbi__get8(s); + pal[i][3] = transp == i ? 0 : 255; + } +} + +static int stbi__gif_header(stbi__context* s, stbi__gif* g, int* comp, int is_info) +{ + stbi_uc version; + if (stbi__get8(s) != 'G' || stbi__get8(s) != 'I' || stbi__get8(s) != 'F' || stbi__get8(s) != '8') + return stbi__err("not GIF", "Corrupt GIF"); + + version = stbi__get8(s); + if (version != '7' && version != '9') + return stbi__err("not GIF", "Corrupt GIF"); + if (stbi__get8(s) != 'a') + return stbi__err("not GIF", "Corrupt GIF"); + + stbi__g_failure_reason = ""; + g->w = stbi__get16le(s); + g->h = stbi__get16le(s); + g->flags = stbi__get8(s); + g->bgindex = stbi__get8(s); + g->ratio = stbi__get8(s); + g->transparent = -1; + + if (g->w > STBI_MAX_DIMENSIONS) + return stbi__err("too large", "Very large image (corrupt?)"); + if (g->h > STBI_MAX_DIMENSIONS) + return stbi__err("too large", "Very large image (corrupt?)"); + + if (comp != 0) + *comp = 4; // can't actually tell whether it's 3 or 4 until we parse the comments + + if (is_info) + return 1; + + if (g->flags & 0x80) + stbi__gif_parse_colortable(s, g->pal, 2 << (g->flags & 7), -1); + + return 1; +} + +static int stbi__gif_info_raw(stbi__context* s, int* x, int* y, int* comp) +{ + stbi__gif* g = (stbi__gif*)stbi__malloc(sizeof(stbi__gif)); + if (!g) + return stbi__err("outofmem", "Out of memory"); + if (!stbi__gif_header(s, g, comp, 1)) { + STBI_FREE(g); + stbi__rewind(s); + return 0; + } + if (x) + *x = g->w; + if (y) + *y = g->h; + STBI_FREE(g); + return 1; +} + +static void stbi__out_gif_code(stbi__gif* g, stbi__uint16 code) +{ + stbi_uc *p, *c; + int idx; + + // recurse to decode the prefixes, since the linked-list is backwards, + // and working backwards through an interleaved image would be nasty + if (g->codes[code].prefix >= 0) + stbi__out_gif_code(g, g->codes[code].prefix); + + if (g->cur_y >= g->max_y) + return; + + idx = g->cur_x + g->cur_y; + p = &g->out[idx]; + g->history[idx / 4] = 1; + + c = &g->color_table[g->codes[code].suffix * 4]; + if (c[3] > 128) { // don't render transparent pixels; + p[0] = c[2]; + p[1] = c[1]; + p[2] = c[0]; + p[3] = c[3]; + } + g->cur_x += 4; + + if (g->cur_x >= g->max_x) { + g->cur_x = g->start_x; + g->cur_y += g->step; + + while (g->cur_y >= g->max_y && g->parse > 0) { + g->step = (1 << g->parse) * g->line_size; + g->cur_y = g->start_y + (g->step >> 1); + --g->parse; + } + } +} + +static stbi_uc* stbi__process_gif_raster(stbi__context* s, stbi__gif* g) +{ + stbi_uc lzw_cs; + stbi__int32 len, init_code; + stbi__uint32 first; + stbi__int32 codesize, codemask, avail, oldcode, bits, valid_bits, clear; + stbi__gif_lzw* p; + + lzw_cs = stbi__get8(s); + if (lzw_cs > 12) + return NULL; + clear = 1 << lzw_cs; + first = 1; + codesize = lzw_cs + 1; + codemask = (1 << codesize) - 1; + bits = 0; + valid_bits = 0; + for (init_code = 0; init_code < clear; init_code++) { + g->codes[init_code].prefix = -1; + g->codes[init_code].first = (stbi_uc)init_code; + g->codes[init_code].suffix = (stbi_uc)init_code; + } + + // support no starting clear code + avail = clear + 2; + oldcode = -1; + + len = 0; + for (;;) { + if (valid_bits < codesize) { + if (len == 0) { + len = stbi__get8(s); // start new block + if (len == 0) + return g->out; + } + --len; + bits |= (stbi__int32)stbi__get8(s) << valid_bits; + valid_bits += 8; + } else { + stbi__int32 code = bits & codemask; + bits >>= codesize; + valid_bits -= codesize; + // @OPTIMIZE: is there some way we can accelerate the non-clear path? + if (code == clear) { // clear code + codesize = lzw_cs + 1; + codemask = (1 << codesize) - 1; + avail = clear + 2; + oldcode = -1; + first = 0; + } else if (code == clear + 1) { // end of stream code + stbi__skip(s, len); + while ((len = stbi__get8(s)) > 0) + stbi__skip(s, len); + return g->out; + } else if (code <= avail) { + if (first) { + return stbi__errpuc("no clear code", "Corrupt GIF"); + } + + if (oldcode >= 0) { + p = &g->codes[avail++]; + if (avail > 8192) { + return stbi__errpuc("too many codes", "Corrupt GIF"); + } + + p->prefix = (stbi__int16)oldcode; + p->first = g->codes[oldcode].first; + p->suffix = (code == avail) ? p->first : g->codes[code].first; + } else if (code == avail) + return stbi__errpuc("illegal code in raster", "Corrupt GIF"); + + stbi__out_gif_code(g, (stbi__uint16)code); + + if ((avail & codemask) == 0 && avail <= 0x0FFF) { + codesize++; + codemask = (1 << codesize) - 1; + } + + oldcode = code; + } else { + return stbi__errpuc("illegal code in raster", "Corrupt GIF"); + } + } + } +} + +// this function is designed to support animated gifs, although stb_image doesn't support it +// two back is the image from two frames ago, used for a very specific disposal format +static stbi_uc* stbi__gif_load_next(stbi__context* s, stbi__gif* g, int* comp, int req_comp, stbi_uc* two_back) +{ + int dispose; + int first_frame; + int pi; + int pcount; + STBI_NOTUSED(req_comp); + + // on first frame, any non-written pixels get the background colour (non-transparent) + first_frame = 0; + if (g->out == 0) { + if (!stbi__gif_header(s, g, comp, 0)) + return 0; // stbi__g_failure_reason set by stbi__gif_header + if (!stbi__mad3sizes_valid(4, g->w, g->h, 0)) + return stbi__errpuc("too large", "GIF image is too large"); + pcount = g->w * g->h; + g->out = (stbi_uc*)stbi__malloc(4 * pcount); + g->background = (stbi_uc*)stbi__malloc(4 * pcount); + g->history = (stbi_uc*)stbi__malloc(pcount); + if (!g->out || !g->background || !g->history) + return stbi__errpuc("outofmem", "Out of memory"); + + // image is treated as "transparent" at the start - ie, nothing overwrites the current background; + // background colour is only used for pixels that are not rendered first frame, after that "background" + // color refers to the color that was there the previous frame. + memset(g->out, 0x00, 4 * pcount); + memset(g->background, 0x00, 4 * pcount); // state of the background (starts transparent) + memset(g->history, 0x00, pcount); // pixels that were affected previous frame + first_frame = 1; + } else { + // second frame - how do we dispose of the previous one? + dispose = (g->eflags & 0x1C) >> 2; + pcount = g->w * g->h; + + if ((dispose == 3) && (two_back == 0)) { + dispose = 2; // if I don't have an image to revert back to, default to the old background + } + + if (dispose == 3) { // use previous graphic + for (pi = 0; pi < pcount; ++pi) { + if (g->history[pi]) { + memcpy(&g->out[pi * 4], &two_back[pi * 4], 4); + } + } + } else if (dispose == 2) { + // restore what was changed last frame to background before that frame; + for (pi = 0; pi < pcount; ++pi) { + if (g->history[pi]) { + memcpy(&g->out[pi * 4], &g->background[pi * 4], 4); + } + } + } else { + // This is a non-disposal case eithe way, so just + // leave the pixels as is, and they will become the new background + // 1: do not dispose + // 0: not specified. + } + + // background is what out is after the undoing of the previou frame; + memcpy(g->background, g->out, 4 * g->w * g->h); + } + + // clear my history; + memset(g->history, 0x00, g->w * g->h); // pixels that were affected previous frame + + for (;;) { + int tag = stbi__get8(s); + switch (tag) { + case 0x2C: /* Image Descriptor */ + { + stbi__int32 x, y, w, h; + stbi_uc* o; + + x = stbi__get16le(s); + y = stbi__get16le(s); + w = stbi__get16le(s); + h = stbi__get16le(s); + if (((x + w) > (g->w)) || ((y + h) > (g->h))) + return stbi__errpuc("bad Image Descriptor", "Corrupt GIF"); + + g->line_size = g->w * 4; + g->start_x = x * 4; + g->start_y = y * g->line_size; + g->max_x = g->start_x + w * 4; + g->max_y = g->start_y + h * g->line_size; + g->cur_x = g->start_x; + g->cur_y = g->start_y; + + // if the width of the specified rectangle is 0, that means + // we may not see *any* pixels or the image is malformed; + // to make sure this is caught, move the current y down to + // max_y (which is what out_gif_code checks). + if (w == 0) + g->cur_y = g->max_y; + + g->lflags = stbi__get8(s); + + if (g->lflags & 0x40) { + g->step = 8 * g->line_size; // first interlaced spacing + g->parse = 3; + } else { + g->step = g->line_size; + g->parse = 0; + } + + if (g->lflags & 0x80) { + stbi__gif_parse_colortable(s, g->lpal, 2 << (g->lflags & 7), g->eflags & 0x01 ? g->transparent : -1); + g->color_table = (stbi_uc*)g->lpal; + } else if (g->flags & 0x80) { + g->color_table = (stbi_uc*)g->pal; + } else + return stbi__errpuc("missing color table", "Corrupt GIF"); + + o = stbi__process_gif_raster(s, g); + if (!o) + return NULL; + + // if this was the first frame, + pcount = g->w * g->h; + if (first_frame && (g->bgindex > 0)) { + // if first frame, any pixel not drawn to gets the background color + for (pi = 0; pi < pcount; ++pi) { + if (g->history[pi] == 0) { + g->pal[g->bgindex][3] = 255; // just in case it was made transparent, undo that; It will be reset next frame if need be; + memcpy(&g->out[pi * 4], &g->pal[g->bgindex], 4); + } + } + } + + return o; + } + + case 0x21: // Comment Extension. + { + int len; + int ext = stbi__get8(s); + if (ext == 0xF9) { // Graphic Control Extension. + len = stbi__get8(s); + if (len == 4) { + g->eflags = stbi__get8(s); + g->delay = 10 * stbi__get16le(s); // delay - 1/100th of a second, saving as 1/1000ths. + + // unset old transparent + if (g->transparent >= 0) { + g->pal[g->transparent][3] = 255; + } + if (g->eflags & 0x01) { + g->transparent = stbi__get8(s); + if (g->transparent >= 0) { + g->pal[g->transparent][3] = 0; + } + } else { + // don't need transparent + stbi__skip(s, 1); + g->transparent = -1; + } + } else { + stbi__skip(s, len); + break; + } + } + while ((len = stbi__get8(s)) != 0) { + stbi__skip(s, len); + } + break; + } + + case 0x3B: // gif stream termination code + return (stbi_uc*)s; // using '1' causes warning on some compilers + + default: + return stbi__errpuc("unknown code", "Corrupt GIF"); + } + } +} + +static void* stbi__load_gif_main_outofmem(stbi__gif* g, stbi_uc* out, int** delays) +{ + STBI_FREE(g->out); + STBI_FREE(g->history); + STBI_FREE(g->background); + + if (out) + STBI_FREE(out); + if (delays && *delays) + STBI_FREE(*delays); + return stbi__errpuc("outofmem", "Out of memory"); +} + +static void* stbi__load_gif_main(stbi__context* s, int** delays, int* x, int* y, int* z, int* comp, int req_comp) +{ + if (stbi__gif_test(s)) { + int layers = 0; + stbi_uc* u = 0; + stbi_uc* out = 0; + stbi_uc* two_back = 0; + stbi__gif g; + int stride; + int out_size = 0; + int delays_size = 0; + + STBI_NOTUSED(out_size); + STBI_NOTUSED(delays_size); + + memset(&g, 0, sizeof(g)); + if (delays) { + *delays = 0; + } + + do { + u = stbi__gif_load_next(s, &g, comp, req_comp, two_back); + if (u == (stbi_uc*)s) + u = 0; // end of animated gif marker + + if (u) { + *x = g.w; + *y = g.h; + ++layers; + stride = g.w * g.h * 4; + + if (out) { + void* tmp = (stbi_uc*)STBI_REALLOC_SIZED(out, out_size, layers * stride); + if (!tmp) + return stbi__load_gif_main_outofmem(&g, out, delays); + else { + out = (stbi_uc*)tmp; + out_size = layers * stride; + } + + if (delays) { + int* new_delays = (int*)STBI_REALLOC_SIZED(*delays, delays_size, sizeof(int) * layers); + if (!new_delays) + return stbi__load_gif_main_outofmem(&g, out, delays); + *delays = new_delays; + delays_size = layers * sizeof(int); + } + } else { + out = (stbi_uc*)stbi__malloc(layers * stride); + if (!out) + return stbi__load_gif_main_outofmem(&g, out, delays); + out_size = layers * stride; + if (delays) { + *delays = (int*)stbi__malloc(layers * sizeof(int)); + if (!*delays) + return stbi__load_gif_main_outofmem(&g, out, delays); + delays_size = layers * sizeof(int); + } + } + memcpy(out + ((layers - 1) * stride), u, stride); + if (layers >= 2) { + two_back = out - 2 * stride; + } + + if (delays) { + (*delays)[layers - 1U] = g.delay; + } + } + } while (u != 0); + + // free temp buffer; + STBI_FREE(g.out); + STBI_FREE(g.history); + STBI_FREE(g.background); + + // do the final conversion after loading everything; + if (req_comp && req_comp != 4) + out = stbi__convert_format(out, 4, req_comp, layers * g.w, g.h); + + *z = layers; + return out; + } else { + return stbi__errpuc("not GIF", "Image was not as a gif type."); + } +} + +static void* stbi__gif_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri) +{ + stbi_uc* u = 0; + stbi__gif g; + memset(&g, 0, sizeof(g)); + STBI_NOTUSED(ri); + + u = stbi__gif_load_next(s, &g, comp, req_comp, 0); + if (u == (stbi_uc*)s) + u = 0; // end of animated gif marker + if (u) { + *x = g.w; + *y = g.h; + + // moved conversion to after successful load so that the same + // can be done for multiple frames. + if (req_comp && req_comp != 4) + u = stbi__convert_format(u, 4, req_comp, g.w, g.h); + } else if (g.out) { + // if there was an error and we allocated an image buffer, free it! + STBI_FREE(g.out); + } + + // free buffers needed for multiple frame loading; + STBI_FREE(g.history); + STBI_FREE(g.background); + + return u; +} + +static int stbi__gif_info(stbi__context* s, int* x, int* y, int* comp) +{ + return stbi__gif_info_raw(s, x, y, comp); +} +# endif + +// ************************************************************************************************* +// Radiance RGBE HDR loader +// originally by Nicolas Schulz +# ifndef STBI_NO_HDR +static int stbi__hdr_test_core(stbi__context* s, char const* signature) +{ + int i; + for (i = 0; signature[i]; ++i) + if (stbi__get8(s) != signature[i]) + return 0; + stbi__rewind(s); + return 1; +} + +static int stbi__hdr_test(stbi__context* s) +{ + int r = stbi__hdr_test_core(s, "#?RADIANCE\n"); + stbi__rewind(s); + if (!r) { + r = stbi__hdr_test_core(s, "#?RGBE\n"); + stbi__rewind(s); + } + return r; +} + +# define STBI__HDR_BUFLEN 1024 +static char* stbi__hdr_gettoken(stbi__context* z, char* buffer) +{ + int len = 0; + char c = '\0'; + + c = (char)stbi__get8(z); + + while (!stbi__at_eof(z) && c != '\n') { + buffer[len++] = c; + if (len == STBI__HDR_BUFLEN - 1) { + // flush to end of line + while (!stbi__at_eof(z) && stbi__get8(z) != '\n') + ; + break; + } + c = (char)stbi__get8(z); + } + + buffer[len] = 0; + return buffer; +} + +static void stbi__hdr_convert(float* output, stbi_uc* input, int req_comp) +{ + if (input[3] != 0) { + float f1; + // Exponent + f1 = (float)ldexp(1.0f, input[3] - (int)(128 + 8)); + if (req_comp <= 2) + output[0] = (input[0] + input[1] + input[2]) * f1 / 3; + else { + output[0] = input[0] * f1; + output[1] = input[1] * f1; + output[2] = input[2] * f1; + } + if (req_comp == 2) + output[1] = 1; + if (req_comp == 4) + output[3] = 1; + } else { + switch (req_comp) { + case 4: + output[3] = 1; /* fallthrough */ + case 3: + output[0] = output[1] = output[2] = 0; + break; + case 2: + output[1] = 1; /* fallthrough */ + case 1: + output[0] = 0; + break; + } + } +} + +static float* stbi__hdr_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri) +{ + char buffer[STBI__HDR_BUFLEN]; + char* token; + int valid = 0; + int width, height; + stbi_uc* scanline; + float* hdr_data; + int len; + unsigned char count, value; + int i, j, k, c1, c2, z; + char const* headerToken; + STBI_NOTUSED(ri); + + // Check identifier + headerToken = stbi__hdr_gettoken(s, buffer); + if (strcmp(headerToken, "#?RADIANCE") != 0 && strcmp(headerToken, "#?RGBE") != 0) + return stbi__errpf("not HDR", "Corrupt HDR image"); + + // Parse header + for (;;) { + token = stbi__hdr_gettoken(s, buffer); + if (token[0] == 0) + break; + if (strcmp(token, "FORMAT=32-bit_rle_rgbe") == 0) + valid = 1; + } + + if (!valid) + return stbi__errpf("unsupported format", "Unsupported HDR format"); + + // Parse width and height + // can't use sscanf() if we're not using stdio! + token = stbi__hdr_gettoken(s, buffer); + if (strncmp(token, "-Y ", 3)) + return stbi__errpf("unsupported data layout", "Unsupported HDR format"); + token += 3; + height = (int)strtol(token, &token, 10); + while (*token == ' ') + ++token; + if (strncmp(token, "+X ", 3)) + return stbi__errpf("unsupported data layout", "Unsupported HDR format"); + token += 3; + width = (int)strtol(token, NULL, 10); + + if (height > STBI_MAX_DIMENSIONS) + return stbi__errpf("too large", "Very large image (corrupt?)"); + if (width > STBI_MAX_DIMENSIONS) + return stbi__errpf("too large", "Very large image (corrupt?)"); + + *x = width; + *y = height; + + if (comp) + *comp = 3; + if (req_comp == 0) + req_comp = 3; + + if (!stbi__mad4sizes_valid(width, height, req_comp, sizeof(float), 0)) + return stbi__errpf("too large", "HDR image is too large"); + + // Read data + hdr_data = (float*)stbi__malloc_mad4(width, height, req_comp, sizeof(float), 0); + if (!hdr_data) + return stbi__errpf("outofmem", "Out of memory"); + + // Load image data + // image data is stored as some number of sca + if (width < 8 || width >= 32768) { + // Read flat data + for (j = 0; j < height; ++j) { + for (i = 0; i < width; ++i) { + stbi_uc rgbe[4]; + main_decode_loop: + stbi__getn(s, rgbe, 4); + stbi__hdr_convert(hdr_data + j * width * req_comp + i * req_comp, rgbe, req_comp); + } + } + } else { + // Read RLE-encoded data + scanline = NULL; + + for (j = 0; j < height; ++j) { + c1 = stbi__get8(s); + c2 = stbi__get8(s); + len = stbi__get8(s); + if (c1 != 2 || c2 != 2 || (len & 0x80)) { + // not run-length encoded, so we have to actually use THIS data as a decoded + // pixel (note this can't be a valid pixel--one of RGB must be >= 128) + stbi_uc rgbe[4]; + rgbe[0] = (stbi_uc)c1; + rgbe[1] = (stbi_uc)c2; + rgbe[2] = (stbi_uc)len; + rgbe[3] = (stbi_uc)stbi__get8(s); + stbi__hdr_convert(hdr_data, rgbe, req_comp); + i = 1; + j = 0; + STBI_FREE(scanline); + goto main_decode_loop; // yes, this makes no sense + } + len <<= 8; + len |= stbi__get8(s); + if (len != width) { + STBI_FREE(hdr_data); + STBI_FREE(scanline); + return stbi__errpf("invalid decoded scanline length", "corrupt HDR"); + } + if (scanline == NULL) { + scanline = (stbi_uc*)stbi__malloc_mad2(width, 4, 0); + if (!scanline) { + STBI_FREE(hdr_data); + return stbi__errpf("outofmem", "Out of memory"); + } + } + + for (k = 0; k < 4; ++k) { + int nleft; + i = 0; + while ((nleft = width - i) > 0) { + count = stbi__get8(s); + if (count > 128) { + // Run + value = stbi__get8(s); + count -= 128; + if ((count == 0) || (count > nleft)) { + STBI_FREE(hdr_data); + STBI_FREE(scanline); + return stbi__errpf("corrupt", "bad RLE data in HDR"); + } + for (z = 0; z < count; ++z) + scanline[i++ * 4 + k] = value; + } else { + // Dump + if ((count == 0) || (count > nleft)) { + STBI_FREE(hdr_data); + STBI_FREE(scanline); + return stbi__errpf("corrupt", "bad RLE data in HDR"); + } + for (z = 0; z < count; ++z) + scanline[i++ * 4 + k] = stbi__get8(s); + } + } + } + for (i = 0; i < width; ++i) + stbi__hdr_convert(hdr_data + (j * width + i) * req_comp, scanline + i * 4, req_comp); + } + if (scanline) + STBI_FREE(scanline); + } + + return hdr_data; +} + +static int stbi__hdr_info(stbi__context* s, int* x, int* y, int* comp) +{ + char buffer[STBI__HDR_BUFLEN]; + char* token; + int valid = 0; + int dummy; + + if (!x) + x = &dummy; + if (!y) + y = &dummy; + if (!comp) + comp = &dummy; + + if (stbi__hdr_test(s) == 0) { + stbi__rewind(s); + return 0; + } + + for (;;) { + token = stbi__hdr_gettoken(s, buffer); + if (token[0] == 0) + break; + if (strcmp(token, "FORMAT=32-bit_rle_rgbe") == 0) + valid = 1; + } + + if (!valid) { + stbi__rewind(s); + return 0; + } + token = stbi__hdr_gettoken(s, buffer); + if (strncmp(token, "-Y ", 3)) { + stbi__rewind(s); + return 0; + } + token += 3; + *y = (int)strtol(token, &token, 10); + while (*token == ' ') + ++token; + if (strncmp(token, "+X ", 3)) { + stbi__rewind(s); + return 0; + } + token += 3; + *x = (int)strtol(token, NULL, 10); + *comp = 3; + return 1; +} +# endif // STBI_NO_HDR + +# ifndef STBI_NO_BMP +static int stbi__bmp_info(stbi__context* s, int* x, int* y, int* comp) +{ + void* p; + stbi__bmp_data info; + + info.all_a = 255; + p = stbi__bmp_parse_header(s, &info); + if (p == NULL) { + stbi__rewind(s); + return 0; + } + if (x) + *x = s->img_x; + if (y) + *y = s->img_y; + if (comp) { + if (info.bpp == 24 && info.ma == 0xff000000) + *comp = 3; + else + *comp = info.ma ? 4 : 3; + } + return 1; +} +# endif + +# ifndef STBI_NO_PSD +static int stbi__psd_info(stbi__context* s, int* x, int* y, int* comp) +{ + int channelCount, dummy, depth; + if (!x) + x = &dummy; + if (!y) + y = &dummy; + if (!comp) + comp = &dummy; + if (stbi__get32be(s) != 0x38425053) { + stbi__rewind(s); + return 0; + } + if (stbi__get16be(s) != 1) { + stbi__rewind(s); + return 0; + } + stbi__skip(s, 6); + channelCount = stbi__get16be(s); + if (channelCount < 0 || channelCount > 16) { + stbi__rewind(s); + return 0; + } + *y = stbi__get32be(s); + *x = stbi__get32be(s); + depth = stbi__get16be(s); + if (depth != 8 && depth != 16) { + stbi__rewind(s); + return 0; + } + if (stbi__get16be(s) != 3) { + stbi__rewind(s); + return 0; + } + *comp = 4; + return 1; +} + +static int stbi__psd_is16(stbi__context* s) +{ + int channelCount, depth; + if (stbi__get32be(s) != 0x38425053) { + stbi__rewind(s); + return 0; + } + if (stbi__get16be(s) != 1) { + stbi__rewind(s); + return 0; + } + stbi__skip(s, 6); + channelCount = stbi__get16be(s); + if (channelCount < 0 || channelCount > 16) { + stbi__rewind(s); + return 0; + } + STBI_NOTUSED(stbi__get32be(s)); + STBI_NOTUSED(stbi__get32be(s)); + depth = stbi__get16be(s); + if (depth != 16) { + stbi__rewind(s); + return 0; + } + return 1; +} +# endif + +# ifndef STBI_NO_PIC +static int stbi__pic_info(stbi__context* s, int* x, int* y, int* comp) +{ + int act_comp = 0, num_packets = 0, chained, dummy; + stbi__pic_packet packets[10]; + + if (!x) + x = &dummy; + if (!y) + y = &dummy; + if (!comp) + comp = &dummy; + + if (!stbi__pic_is4(s, "\x53\x80\xF6\x34")) { + stbi__rewind(s); + return 0; + } + + stbi__skip(s, 88); + + *x = stbi__get16be(s); + *y = stbi__get16be(s); + if (stbi__at_eof(s)) { + stbi__rewind(s); + return 0; + } + if ((*x) != 0 && (1 << 28) / (*x) < (*y)) { + stbi__rewind(s); + return 0; + } + + stbi__skip(s, 8); + + do { + stbi__pic_packet* packet; + + if (num_packets == sizeof(packets) / sizeof(packets[0])) + return 0; + + packet = &packets[num_packets++]; + chained = stbi__get8(s); + packet->size = stbi__get8(s); + packet->type = stbi__get8(s); + packet->channel = stbi__get8(s); + act_comp |= packet->channel; + + if (stbi__at_eof(s)) { + stbi__rewind(s); + return 0; + } + if (packet->size != 8) { + stbi__rewind(s); + return 0; + } + } while (chained); + + *comp = (act_comp & 0x10 ? 4 : 3); + + return 1; +} +# endif + +// ************************************************************************************************* +// Portable Gray Map and Portable Pixel Map loader +// by Ken Miller +// +// PGM: http://netpbm.sourceforge.net/doc/pgm.html +// PPM: http://netpbm.sourceforge.net/doc/ppm.html +// +// Known limitations: +// Does not support comments in the header section +// Does not support ASCII image data (formats P2 and P3) + +# ifndef STBI_NO_PNM + +static int stbi__pnm_test(stbi__context* s) +{ + char p, t; + p = (char)stbi__get8(s); + t = (char)stbi__get8(s); + if (p != 'P' || (t != '5' && t != '6')) { + stbi__rewind(s); + return 0; + } + return 1; +} + +static void* stbi__pnm_load(stbi__context* s, int* x, int* y, int* comp, int req_comp, stbi__result_info* ri) +{ + stbi_uc* out; + STBI_NOTUSED(ri); + + ri->bits_per_channel = stbi__pnm_info(s, (int*)&s->img_x, (int*)&s->img_y, (int*)&s->img_n); + if (ri->bits_per_channel == 0) + return 0; + + if (s->img_y > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + if (s->img_x > STBI_MAX_DIMENSIONS) + return stbi__errpuc("too large", "Very large image (corrupt?)"); + + *x = s->img_x; + *y = s->img_y; + if (comp) + *comp = s->img_n; + + if (!stbi__mad4sizes_valid(s->img_n, s->img_x, s->img_y, ri->bits_per_channel / 8, 0)) + return stbi__errpuc("too large", "PNM too large"); + + out = (stbi_uc*)stbi__malloc_mad4(s->img_n, s->img_x, s->img_y, ri->bits_per_channel / 8, 0); + if (!out) + return stbi__errpuc("outofmem", "Out of memory"); + if (!stbi__getn(s, out, s->img_n * s->img_x * s->img_y * (ri->bits_per_channel / 8))) { + STBI_FREE(out); + return stbi__errpuc("bad PNM", "PNM file truncated"); + } + + if (req_comp && req_comp != s->img_n) { + if (ri->bits_per_channel == 16) { + out = (stbi_uc*)stbi__convert_format16((stbi__uint16*)out, s->img_n, req_comp, s->img_x, s->img_y); + } else { + out = stbi__convert_format(out, s->img_n, req_comp, s->img_x, s->img_y); + } + if (out == NULL) + return out; // stbi__convert_format frees input on failure + } + return out; +} + +static int stbi__pnm_isspace(char c) +{ + return c == ' ' || c == '\t' || c == '\n' || c == '\v' || c == '\f' || c == '\r'; +} + +static void stbi__pnm_skip_whitespace(stbi__context* s, char* c) +{ + for (;;) { + while (!stbi__at_eof(s) && stbi__pnm_isspace(*c)) + *c = (char)stbi__get8(s); + + if (stbi__at_eof(s) || *c != '#') + break; + + while (!stbi__at_eof(s) && *c != '\n' && *c != '\r') + *c = (char)stbi__get8(s); + } +} + +static int stbi__pnm_isdigit(char c) +{ + return c >= '0' && c <= '9'; +} + +static int stbi__pnm_getinteger(stbi__context* s, char* c) +{ + int value = 0; + + while (!stbi__at_eof(s) && stbi__pnm_isdigit(*c)) { + value = value * 10 + (*c - '0'); + *c = (char)stbi__get8(s); + if ((value > 214748364) || (value == 214748364 && *c > '7')) + return stbi__err("integer parse overflow", "Parsing an integer in the PPM header overflowed a 32-bit int"); + } + + return value; +} + +static int stbi__pnm_info(stbi__context* s, int* x, int* y, int* comp) +{ + int maxv, dummy; + char c, p, t; + + if (!x) + x = &dummy; + if (!y) + y = &dummy; + if (!comp) + comp = &dummy; + + stbi__rewind(s); + + // Get identifier + p = (char)stbi__get8(s); + t = (char)stbi__get8(s); + if (p != 'P' || (t != '5' && t != '6')) { + stbi__rewind(s); + return 0; + } + + *comp = (t == '6') ? 3 : 1; // '5' is 1-component .pgm; '6' is 3-component .ppm + + c = (char)stbi__get8(s); + stbi__pnm_skip_whitespace(s, &c); + + *x = stbi__pnm_getinteger(s, &c); // read width + if (*x == 0) + return stbi__err("invalid width", "PPM image header had zero or overflowing width"); + stbi__pnm_skip_whitespace(s, &c); + + *y = stbi__pnm_getinteger(s, &c); // read height + if (*y == 0) + return stbi__err("invalid width", "PPM image header had zero or overflowing width"); + stbi__pnm_skip_whitespace(s, &c); + + maxv = stbi__pnm_getinteger(s, &c); // read max value + if (maxv > 65535) + return stbi__err("max value > 65535", "PPM image supports only 8-bit and 16-bit images"); + else if (maxv > 255) + return 16; + else + return 8; +} + +static int stbi__pnm_is16(stbi__context* s) +{ + if (stbi__pnm_info(s, NULL, NULL, NULL) == 16) + return 1; + return 0; +} +# endif + +static int stbi__info_main(stbi__context* s, int* x, int* y, int* comp) +{ +# ifndef STBI_NO_JPEG + if (stbi__jpeg_info(s, x, y, comp)) + return 1; +# endif + +# ifndef STBI_NO_PNG + if (stbi__png_info(s, x, y, comp)) + return 1; +# endif + +# ifndef STBI_NO_GIF + if (stbi__gif_info(s, x, y, comp)) + return 1; +# endif + +# ifndef STBI_NO_BMP + if (stbi__bmp_info(s, x, y, comp)) + return 1; +# endif + +# ifndef STBI_NO_PSD + if (stbi__psd_info(s, x, y, comp)) + return 1; +# endif + +# ifndef STBI_NO_PIC + if (stbi__pic_info(s, x, y, comp)) + return 1; +# endif + +# ifndef STBI_NO_PNM + if (stbi__pnm_info(s, x, y, comp)) + return 1; +# endif + +# ifndef STBI_NO_HDR + if (stbi__hdr_info(s, x, y, comp)) + return 1; +# endif + +// test tga last because it's a crappy test! +# ifndef STBI_NO_TGA + if (stbi__tga_info(s, x, y, comp)) + return 1; +# endif + return stbi__err("unknown image type", "Image not of any known type, or corrupt"); +} + +static int stbi__is_16_main(stbi__context* s) +{ +# ifndef STBI_NO_PNG + if (stbi__png_is16(s)) + return 1; +# endif + +# ifndef STBI_NO_PSD + if (stbi__psd_is16(s)) + return 1; +# endif + +# ifndef STBI_NO_PNM + if (stbi__pnm_is16(s)) + return 1; +# endif + return 0; +} + +# ifndef STBI_NO_STDIO +STBIDEF int stbi_info(char const* filename, int* x, int* y, int* comp) +{ + FILE* f = stbi__fopen(filename, "rb"); + int result; + if (!f) + return stbi__err("can't fopen", "Unable to open file"); + result = stbi_info_from_file(f, x, y, comp); + fclose(f); + return result; +} + +STBIDEF int stbi_info_from_file(FILE* f, int* x, int* y, int* comp) +{ + int r; + stbi__context s; + long pos = ftell(f); + stbi__start_file(&s, f); + r = stbi__info_main(&s, x, y, comp); + fseek(f, pos, SEEK_SET); + return r; +} + +STBIDEF int stbi_is_16_bit(char const* filename) +{ + FILE* f = stbi__fopen(filename, "rb"); + int result; + if (!f) + return stbi__err("can't fopen", "Unable to open file"); + result = stbi_is_16_bit_from_file(f); + fclose(f); + return result; +} + +STBIDEF int stbi_is_16_bit_from_file(FILE* f) +{ + int r; + stbi__context s; + long pos = ftell(f); + stbi__start_file(&s, f); + r = stbi__is_16_main(&s); + fseek(f, pos, SEEK_SET); + return r; +} +# endif // !STBI_NO_STDIO + +STBIDEF int stbi_info_from_memory(stbi_uc const* buffer, int len, int* x, int* y, int* comp) +{ + stbi__context s; + stbi__start_mem(&s, buffer, len); + return stbi__info_main(&s, x, y, comp); +} + +STBIDEF int stbi_info_from_callbacks(stbi_io_callbacks const* c, void* user, int* x, int* y, int* comp) +{ + stbi__context s; + stbi__start_callbacks(&s, (stbi_io_callbacks*)c, user); + return stbi__info_main(&s, x, y, comp); +} + +STBIDEF int stbi_is_16_bit_from_memory(stbi_uc const* buffer, int len) +{ + stbi__context s; + stbi__start_mem(&s, buffer, len); + return stbi__is_16_main(&s); +} + +STBIDEF int stbi_is_16_bit_from_callbacks(stbi_io_callbacks const* c, void* user) +{ + stbi__context s; + stbi__start_callbacks(&s, (stbi_io_callbacks*)c, user); + return stbi__is_16_main(&s); +} + +#endif // STB_IMAGE_IMPLEMENTATION + +/* + revision history: + 2.20 (2019-02-07) support utf8 filenames in Windows; fix warnings and platform ifdefs + 2.19 (2018-02-11) fix warning + 2.18 (2018-01-30) fix warnings + 2.17 (2018-01-29) change sbti__shiftsigned to avoid clang -O2 bug + 1-bit BMP + *_is_16_bit api + avoid warnings + 2.16 (2017-07-23) all functions have 16-bit variants; + STBI_NO_STDIO works again; + compilation fixes; + fix rounding in unpremultiply; + optimize vertical flip; + disable raw_len validation; + documentation fixes + 2.15 (2017-03-18) fix png-1,2,4 bug; now all Imagenet JPGs decode; + warning fixes; disable run-time SSE detection on gcc; + uniform handling of optional "return" values; + thread-safe initialization of zlib tables + 2.14 (2017-03-03) remove deprecated STBI_JPEG_OLD; fixes for Imagenet JPGs + 2.13 (2016-11-29) add 16-bit API, only supported for PNG right now + 2.12 (2016-04-02) fix typo in 2.11 PSD fix that caused crashes + 2.11 (2016-04-02) allocate large structures on the stack + remove white matting for transparent PSD + fix reported channel count for PNG & BMP + re-enable SSE2 in non-gcc 64-bit + support RGB-formatted JPEG + read 16-bit PNGs (only as 8-bit) + 2.10 (2016-01-22) avoid warning introduced in 2.09 by STBI_REALLOC_SIZED + 2.09 (2016-01-16) allow comments in PNM files + 16-bit-per-pixel TGA (not bit-per-component) + info() for TGA could break due to .hdr handling + info() for BMP to shares code instead of sloppy parse + can use STBI_REALLOC_SIZED if allocator doesn't support realloc + code cleanup + 2.08 (2015-09-13) fix to 2.07 cleanup, reading RGB PSD as RGBA + 2.07 (2015-09-13) fix compiler warnings + partial animated GIF support + limited 16-bpc PSD support + #ifdef unused functions + bug with < 92 byte PIC,PNM,HDR,TGA + 2.06 (2015-04-19) fix bug where PSD returns wrong '*comp' value + 2.05 (2015-04-19) fix bug in progressive JPEG handling, fix warning + 2.04 (2015-04-15) try to re-enable SIMD on MinGW 64-bit + 2.03 (2015-04-12) extra corruption checking (mmozeiko) + stbi_set_flip_vertically_on_load (nguillemot) + fix NEON support; fix mingw support + 2.02 (2015-01-19) fix incorrect assert, fix warning + 2.01 (2015-01-17) fix various warnings; suppress SIMD on gcc 32-bit without -msse2 + 2.00b (2014-12-25) fix STBI_MALLOC in progressive JPEG + 2.00 (2014-12-25) optimize JPG, including x86 SSE2 & NEON SIMD (ryg) + progressive JPEG (stb) + PGM/PPM support (Ken Miller) + STBI_MALLOC,STBI_REALLOC,STBI_FREE + GIF bugfix -- seemingly never worked + STBI_NO_*, STBI_ONLY_* + 1.48 (2014-12-14) fix incorrectly-named assert() + 1.47 (2014-12-14) 1/2/4-bit PNG support, both direct and paletted (Omar Cornut & stb) + optimize PNG (ryg) + fix bug in interlaced PNG with user-specified channel count (stb) + 1.46 (2014-08-26) + fix broken tRNS chunk (colorkey-style transparency) in non-paletted PNG + 1.45 (2014-08-16) + fix MSVC-ARM internal compiler error by wrapping malloc + 1.44 (2014-08-07) + various warning fixes from Ronny Chevalier + 1.43 (2014-07-15) + fix MSVC-only compiler problem in code changed in 1.42 + 1.42 (2014-07-09) + don't define _CRT_SECURE_NO_WARNINGS (affects user code) + fixes to stbi__cleanup_jpeg path + added STBI_ASSERT to avoid requiring assert.h + 1.41 (2014-06-25) + fix search&replace from 1.36 that messed up comments/error messages + 1.40 (2014-06-22) + fix gcc struct-initialization warning + 1.39 (2014-06-15) + fix to TGA optimization when req_comp != number of components in TGA; + fix to GIF loading because BMP wasn't rewinding (whoops, no GIFs in my test suite) + add support for BMP version 5 (more ignored fields) + 1.38 (2014-06-06) + suppress MSVC warnings on integer casts truncating values + fix accidental rename of 'skip' field of I/O + 1.37 (2014-06-04) + remove duplicate typedef + 1.36 (2014-06-03) + convert to header file single-file library + if de-iphone isn't set, load iphone images color-swapped instead of returning NULL + 1.35 (2014-05-27) + various warnings + fix broken STBI_SIMD path + fix bug where stbi_load_from_file no longer left file pointer in correct place + fix broken non-easy path for 32-bit BMP (possibly never used) + TGA optimization by Arseny Kapoulkine + 1.34 (unknown) + use STBI_NOTUSED in stbi__resample_row_generic(), fix one more leak in tga failure case + 1.33 (2011-07-14) + make stbi_is_hdr work in STBI_NO_HDR (as specified), minor compiler-friendly improvements + 1.32 (2011-07-13) + support for "info" function for all supported filetypes (SpartanJ) + 1.31 (2011-06-20) + a few more leak fixes, bug in PNG handling (SpartanJ) + 1.30 (2011-06-11) + added ability to load files via callbacks to accomidate custom input streams (Ben Wenger) + removed deprecated format-specific test/load functions + removed support for installable file formats (stbi_loader) -- would have been broken for IO callbacks anyway + error cases in bmp and tga give messages and don't leak (Raymond Barbiero, grisha) + fix inefficiency in decoding 32-bit BMP (David Woo) + 1.29 (2010-08-16) + various warning fixes from Aurelien Pocheville + 1.28 (2010-08-01) + fix bug in GIF palette transparency (SpartanJ) + 1.27 (2010-08-01) + cast-to-stbi_uc to fix warnings + 1.26 (2010-07-24) + fix bug in file buffering for PNG reported by SpartanJ + 1.25 (2010-07-17) + refix trans_data warning (Won Chun) + 1.24 (2010-07-12) + perf improvements reading from files on platforms with lock-heavy fgetc() + minor perf improvements for jpeg + deprecated type-specific functions so we'll get feedback if they're needed + attempt to fix trans_data warning (Won Chun) + 1.23 fixed bug in iPhone support + 1.22 (2010-07-10) + removed image *writing* support + stbi_info support from Jetro Lauha + GIF support from Jean-Marc Lienher + iPhone PNG-extensions from James Brown + warning-fixes from Nicolas Schulz and Janez Zemva (i.stbi__err. Janez (U+017D)emva) + 1.21 fix use of 'stbi_uc' in header (reported by jon blow) + 1.20 added support for Softimage PIC, by Tom Seddon + 1.19 bug in interlaced PNG corruption check (found by ryg) + 1.18 (2008-08-02) + fix a threading bug (local mutable static) + 1.17 support interlaced PNG + 1.16 major bugfix - stbi__convert_format converted one too many pixels + 1.15 initialize some fields for thread safety + 1.14 fix threadsafe conversion bug + header-file-only version (#define STBI_HEADER_FILE_ONLY before including) + 1.13 threadsafe + 1.12 const qualifiers in the API + 1.11 Support installable IDCT, colorspace conversion routines + 1.10 Fixes for 64-bit (don't use "unsigned long") + optimized upsampling by Fabian "ryg" Giesen + 1.09 Fix format-conversion for PSD code (bad global variables!) + 1.08 Thatcher Ulrich's PSD code integrated by Nicolas Schulz + 1.07 attempt to fix C++ warning/errors again + 1.06 attempt to fix C++ warning/errors again + 1.05 fix TGA loading to return correct *comp and use good luminance calc + 1.04 default float alpha is 1, not 255; use 'void *' for stbi_image_free + 1.03 bugfixes to STBI_NO_STDIO, STBI_NO_HDR + 1.02 support for (subset of) HDR files, float interface for preferred access to them + 1.01 fix bug: possible bug in handling right-side up bmps... not sure + fix bug: the stbi__bmp_load() and stbi__tga_load() functions didn't work at all + 1.00 interface to zlib that skips zlib header + 0.99 correct handling of alpha in palette + 0.98 TGA loader by lonesock; dynamically add loaders (untested) + 0.97 jpeg errors on too large a file; also catch another malloc failure + 0.96 fix detection of invalid v value - particleman@mollyrocket forum + 0.95 during header scan, seek to markers in case of padding + 0.94 STBI_NO_STDIO to disable stdio usage; rename all #defines the same + 0.93 handle jpegtran output; verbose errors + 0.92 read 4,8,16,24,32-bit BMP files of several formats + 0.91 output 24-bit Windows 3.0 BMP files + 0.90 fix a few more warnings; bump version number to approach 1.0 + 0.61 bugfixes due to Marc LeBlanc, Christopher Lloyd + 0.60 fix compiling as c++ + 0.59 fix warnings: merge Dave Moore's -Wall fixes + 0.58 fix bug: zlib uncompressed mode len/nlen was wrong endian + 0.57 fix bug: jpg last huffman symbol before marker was >9 bits but less than 16 available + 0.56 fix bug: zlib uncompressed mode len vs. nlen + 0.55 fix bug: restart_interval not initialized to 0 + 0.54 allow NULL for 'int *comp' + 0.53 fix bug in png 3->4; speedup png decoding + 0.52 png handles req_comp=3,4 directly; minor cleanup; jpeg comments + 0.51 obey req_comp requests, 1-component jpegs return as 1-component, + on 'test' only check type, not whether we support this variant + 0.50 (2006-11-19) + first released version +*/ + +/* +------------------------------------------------------------------------------ +This software is available under 2 licenses -- choose whichever you prefer. +------------------------------------------------------------------------------ +ALTERNATIVE A - MIT License +Copyright (c) 2017 Sean Barrett +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +of the Software, and to permit persons to whom the Software is furnished to do +so, subject to the following conditions: +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. +------------------------------------------------------------------------------ +ALTERNATIVE B - Public Domain (www.unlicense.org) +This is free and unencumbered software released into the public domain. +Anyone is free to copy, modify, publish, use, compile, sell, or distribute this +software, either in source code form or as a compiled binary, for any purpose, +commercial or non-commercial, and by any means. +In jurisdictions that recognize copyright laws, the author or authors of this +software dedicate any and all copyright interest in the software to the public +domain. We make this dedication for the benefit of the public at large and to +the detriment of our heirs and successors. We intend this dedication to be an +overt act of relinquishment in perpetuity of all present and future rights to +this software under copyright law. +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN +ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION +WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +------------------------------------------------------------------------------ +*/ diff --git a/src/image/stb_image_write.h b/src/image/stb_image_write.h new file mode 100644 index 0000000..6ed3dd8 --- /dev/null +++ b/src/image/stb_image_write.h @@ -0,0 +1,1807 @@ +/* stb_image_write - v1.16 - public domain - http://nothings.org/stb + writes out PNG/BMP/TGA/JPEG/HDR images to C stdio - Sean Barrett 2010-2015 + no warranty implied; use at your own risk + + Before #including, + + #define STB_IMAGE_WRITE_IMPLEMENTATION + + in the file that you want to have the implementation. + + Will probably not work correctly with strict-aliasing optimizations. + +ABOUT: + + This header file is a library for writing images to C stdio or a callback. + + The PNG output is not optimal; it is 20-50% larger than the file + written by a decent optimizing implementation; though providing a custom + zlib compress function (see STBIW_ZLIB_COMPRESS) can mitigate that. + This library is designed for source code compactness and simplicity, + not optimal image file size or run-time performance. + +BUILDING: + + You can #define STBIW_ASSERT(x) before the #include to avoid using assert.h. + You can #define STBIW_MALLOC(), STBIW_REALLOC(), and STBIW_FREE() to replace + malloc,realloc,free. + You can #define STBIW_MEMMOVE() to replace memmove() + You can #define STBIW_ZLIB_COMPRESS to use a custom zlib-style compress function + for PNG compression (instead of the builtin one), it must have the following signature: + unsigned char * my_compress(unsigned char *data, int data_len, int *out_len, int quality); + The returned data will be freed with STBIW_FREE() (free() by default), + so it must be heap allocated with STBIW_MALLOC() (malloc() by default), + +UNICODE: + + If compiling for Windows and you wish to use Unicode filenames, compile + with + #define STBIW_WINDOWS_UTF8 + and pass utf8-encoded filenames. Call stbiw_convert_wchar_to_utf8 to convert + Windows wchar_t filenames to utf8. + +USAGE: + + There are five functions, one for each image file format: + + int stbi_write_png(char const *filename, int w, int h, int comp, const void *data, int stride_in_bytes); + int stbi_write_bmp(char const *filename, int w, int h, int comp, const void *data); + int stbi_write_tga(char const *filename, int w, int h, int comp, const void *data); + int stbi_write_jpg(char const *filename, int w, int h, int comp, const void *data, int quality); + int stbi_write_hdr(char const *filename, int w, int h, int comp, const float *data); + + void stbi_flip_vertically_on_write(int flag); // flag is non-zero to flip data vertically + + There are also five equivalent functions that use an arbitrary write function. You are + expected to open/close your file-equivalent before and after calling these: + + int stbi_write_png_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data, int stride_in_bytes); + int stbi_write_bmp_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data); + int stbi_write_tga_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data); + int stbi_write_hdr_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const float *data); + int stbi_write_jpg_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data, int quality); + + where the callback is: + void stbi_write_func(void *context, void *data, int size); + + You can configure it with these global variables: + int stbi_write_tga_with_rle; // defaults to true; set to 0 to disable RLE + int stbi_write_png_compression_level; // defaults to 8; set to higher for more compression + int stbi_write_force_png_filter; // defaults to -1; set to 0..5 to force a filter mode + + + You can define STBI_WRITE_NO_STDIO to disable the file variant of these + functions, so the library will not use stdio.h at all. However, this will + also disable HDR writing, because it requires stdio for formatted output. + + Each function returns 0 on failure and non-0 on success. + + The functions create an image file defined by the parameters. The image + is a rectangle of pixels stored from left-to-right, top-to-bottom. + Each pixel contains 'comp' channels of data stored interleaved with 8-bits + per channel, in the following order: 1=Y, 2=YA, 3=RGB, 4=RGBA. (Y is + monochrome color.) The rectangle is 'w' pixels wide and 'h' pixels tall. + The *data pointer points to the first byte of the top-left-most pixel. + For PNG, "stride_in_bytes" is the distance in bytes from the first byte of + a row of pixels to the first byte of the next row of pixels. + + PNG creates output files with the same number of components as the input. + The BMP format expands Y to RGB in the file format and does not + output alpha. + + PNG supports writing rectangles of data even when the bytes storing rows of + data are not consecutive in memory (e.g. sub-rectangles of a larger image), + by supplying the stride between the beginning of adjacent rows. The other + formats do not. (Thus you cannot write a native-format BMP through the BMP + writer, both because it is in BGR order and because it may have padding + at the end of the line.) + + PNG allows you to set the deflate compression level by setting the global + variable 'stbi_write_png_compression_level' (it defaults to 8). + + HDR expects linear float data. Since the format is always 32-bit rgb(e) + data, alpha (if provided) is discarded, and for monochrome data it is + replicated across all three channels. + + TGA supports RLE or non-RLE compressed data. To use non-RLE-compressed + data, set the global variable 'stbi_write_tga_with_rle' to 0. + + JPEG does ignore alpha channels in input data; quality is between 1 and 100. + Higher quality looks better but results in a bigger image. + JPEG baseline (no JPEG progressive). + +CREDITS: + + + Sean Barrett - PNG/BMP/TGA + Baldur Karlsson - HDR + Jean-Sebastien Guay - TGA monochrome + Tim Kelsey - misc enhancements + Alan Hickman - TGA RLE + Emmanuel Julien - initial file IO callback implementation + Jon Olick - original jo_jpeg.cpp code + Daniel Gibson - integrate JPEG, allow external zlib + Aarni Koskela - allow choosing PNG filter + + bugfixes: + github:Chribba + Guillaume Chereau + github:jry2 + github:romigrou + Sergio Gonzalez + Jonas Karlsson + Filip Wasil + Thatcher Ulrich + github:poppolopoppo + Patrick Boettcher + github:xeekworx + Cap Petschulat + Simon Rodriguez + Ivan Tikhonov + github:ignotion + Adam Schackart + Andrew Kensler + +LICENSE + + See end of file for license information. + +*/ + +#ifndef INCLUDE_STB_IMAGE_WRITE_H +# define INCLUDE_STB_IMAGE_WRITE_H + +# include + +// if STB_IMAGE_WRITE_STATIC causes problems, try defining STBIWDEF to 'inline' or 'static inline' +# ifndef STBIWDEF +# ifdef STB_IMAGE_WRITE_STATIC +# define STBIWDEF static +# else +# ifdef __cplusplus +# define STBIWDEF extern "C" +# else +# define STBIWDEF extern +# endif +# endif +# endif + +# ifndef STB_IMAGE_WRITE_STATIC // C++ forbids static forward declarations +STBIWDEF int stbi_write_tga_with_rle; +STBIWDEF int stbi_write_png_compression_level; +STBIWDEF int stbi_write_force_png_filter; +# endif + +# ifndef STBI_WRITE_NO_STDIO +STBIWDEF int stbi_write_png(char const* filename, int w, int h, int comp, void const* data, int stride_in_bytes); +STBIWDEF int stbi_write_bmp(char const* filename, int w, int h, int comp, void const* data); +STBIWDEF int stbi_write_tga(char const* filename, int w, int h, int comp, void const* data); +STBIWDEF int stbi_write_hdr(char const* filename, int w, int h, int comp, float const* data); +STBIWDEF int stbi_write_jpg(char const* filename, int x, int y, int comp, void const* data, int quality); + +# ifdef STBIW_WINDOWS_UTF8 +STBIWDEF int stbiw_convert_wchar_to_utf8(char* buffer, size_t bufferlen, wchar_t const* input); +# endif +# endif + +typedef void stbi_write_func(void* context, void* data, int size); + +STBIWDEF int stbi_write_png_to_func(stbi_write_func* func, void* context, int w, int h, int comp, void const* data, int stride_in_bytes); +STBIWDEF int stbi_write_bmp_to_func(stbi_write_func* func, void* context, int w, int h, int comp, void const* data); +STBIWDEF int stbi_write_tga_to_func(stbi_write_func* func, void* context, int w, int h, int comp, void const* data); +STBIWDEF int stbi_write_hdr_to_func(stbi_write_func* func, void* context, int w, int h, int comp, float const* data); +STBIWDEF int stbi_write_jpg_to_func(stbi_write_func* func, void* context, int x, int y, int comp, void const* data, int quality); + +STBIWDEF void stbi_flip_vertically_on_write(int flip_boolean); + +#endif // INCLUDE_STB_IMAGE_WRITE_H + +#ifdef STB_IMAGE_WRITE_IMPLEMENTATION + +# ifdef _WIN32 +# ifndef _CRT_SECURE_NO_WARNINGS +# define _CRT_SECURE_NO_WARNINGS +# endif +# ifndef _CRT_NONSTDC_NO_DEPRECATE +# define _CRT_NONSTDC_NO_DEPRECATE +# endif +# endif + +# ifndef STBI_WRITE_NO_STDIO +# include +# endif // STBI_WRITE_NO_STDIO + +# include +# include +# include +# include + +# if defined(STBIW_MALLOC) && defined(STBIW_FREE) && (defined(STBIW_REALLOC) || defined(STBIW_REALLOC_SIZED)) +// ok +# elif !defined(STBIW_MALLOC) && !defined(STBIW_FREE) && !defined(STBIW_REALLOC) && !defined(STBIW_REALLOC_SIZED) +// ok +# else +# error "Must define all or none of STBIW_MALLOC, STBIW_FREE, and STBIW_REALLOC (or STBIW_REALLOC_SIZED)." +# endif + +# ifndef STBIW_MALLOC +# define STBIW_MALLOC(sz) malloc(sz) +# define STBIW_REALLOC(p, newsz) realloc(p, newsz) +# define STBIW_FREE(p) free(p) +# endif + +# ifndef STBIW_REALLOC_SIZED +# define STBIW_REALLOC_SIZED(p, oldsz, newsz) STBIW_REALLOC(p, newsz) +# endif + +# ifndef STBIW_MEMMOVE +# define STBIW_MEMMOVE(a, b, sz) memmove(a, b, sz) +# endif + +# ifndef STBIW_ASSERT +# include +# define STBIW_ASSERT(x) assert(x) +# endif + +# define STBIW_UCHAR(x) (unsigned char)((x) & 0xff) + +# ifdef STB_IMAGE_WRITE_STATIC +static int stbi_write_png_compression_level = 8; +static int stbi_write_tga_with_rle = 1; +static int stbi_write_force_png_filter = -1; +# else +int stbi_write_png_compression_level = 8; +int stbi_write_tga_with_rle = 1; +int stbi_write_force_png_filter = -1; +# endif + +static int stbi__flip_vertically_on_write = 0; + +STBIWDEF void stbi_flip_vertically_on_write(int flag) +{ + stbi__flip_vertically_on_write = flag; +} + +typedef struct +{ + stbi_write_func* func; + void* context; + unsigned char buffer[64]; + int buf_used; +} stbi__write_context; + +// initialize a callback-based context +static void stbi__start_write_callbacks(stbi__write_context* s, stbi_write_func* c, void* context) +{ + s->func = c; + s->context = context; +} + +# ifndef STBI_WRITE_NO_STDIO + +static void stbi__stdio_write(void* context, void* data, int size) +{ + fwrite(data, 1, size, (FILE*)context); +} + +# if defined(_WIN32) && defined(STBIW_WINDOWS_UTF8) +# ifdef __cplusplus +# define STBIW_EXTERN extern "C" +# else +# define STBIW_EXTERN extern +# endif +STBIW_EXTERN __declspec(dllimport) int __stdcall MultiByteToWideChar(unsigned int cp, unsigned long flags, char const* str, int cbmb, wchar_t* widestr, int cchwide); +STBIW_EXTERN __declspec(dllimport) int __stdcall WideCharToMultiByte(unsigned int cp, unsigned long flags, wchar_t const* widestr, int cchwide, char* str, int cbmb, char const* defchar, int* used_default); + +STBIWDEF int stbiw_convert_wchar_to_utf8(char* buffer, size_t bufferlen, wchar_t const* input) +{ + return WideCharToMultiByte(65001 /* UTF8 */, 0, input, -1, buffer, (int)bufferlen, NULL, NULL); +} +# endif + +static FILE* stbiw__fopen(char const* filename, char const* mode) +{ + FILE* f; +# if defined(_WIN32) && defined(STBIW_WINDOWS_UTF8) + wchar_t wMode[64]; + wchar_t wFilename[1024]; + if (0 == MultiByteToWideChar(65001 /* UTF8 */, 0, filename, -1, wFilename, sizeof(wFilename) / sizeof(*wFilename))) + return 0; + + if (0 == MultiByteToWideChar(65001 /* UTF8 */, 0, mode, -1, wMode, sizeof(wMode) / sizeof(*wMode))) + return 0; + +# if defined(_MSC_VER) && _MSC_VER >= 1400 + if (0 != _wfopen_s(&f, wFilename, wMode)) + f = 0; +# else + f = _wfopen(wFilename, wMode); +# endif + +# elif defined(_MSC_VER) && _MSC_VER >= 1400 + if (0 != fopen_s(&f, filename, mode)) + f = 0; +# else + f = fopen(filename, mode); +# endif + return f; +} + +static int stbi__start_write_file(stbi__write_context* s, char const* filename) +{ + FILE* f = stbiw__fopen(filename, "wb"); + stbi__start_write_callbacks(s, stbi__stdio_write, (void*)f); + return f != NULL; +} + +static void stbi__end_write_file(stbi__write_context* s) +{ + fclose((FILE*)s->context); +} + +# endif // !STBI_WRITE_NO_STDIO + +typedef unsigned int stbiw_uint32; +typedef int stb_image_write_test[sizeof(stbiw_uint32) == 4 ? 1 : -1]; + +static void stbiw__writefv(stbi__write_context* s, char const* fmt, va_list v) +{ + while (*fmt) { + switch (*fmt++) { + case ' ': + break; + case '1': { + unsigned char x = STBIW_UCHAR(va_arg(v, int)); + s->func(s->context, &x, 1); + break; + } + case '2': { + int x = va_arg(v, int); + unsigned char b[2]; + b[0] = STBIW_UCHAR(x); + b[1] = STBIW_UCHAR(x >> 8); + s->func(s->context, b, 2); + break; + } + case '4': { + stbiw_uint32 x = va_arg(v, int); + unsigned char b[4]; + b[0] = STBIW_UCHAR(x); + b[1] = STBIW_UCHAR(x >> 8); + b[2] = STBIW_UCHAR(x >> 16); + b[3] = STBIW_UCHAR(x >> 24); + s->func(s->context, b, 4); + break; + } + default: + STBIW_ASSERT(0); + return; + } + } +} + +static void stbiw__writef(stbi__write_context* s, char const* fmt, ...) +{ + va_list v; + va_start(v, fmt); + stbiw__writefv(s, fmt, v); + va_end(v); +} + +static void stbiw__write_flush(stbi__write_context* s) +{ + if (s->buf_used) { + s->func(s->context, &s->buffer, s->buf_used); + s->buf_used = 0; + } +} + +static void stbiw__putc(stbi__write_context* s, unsigned char c) +{ + s->func(s->context, &c, 1); +} + +static void stbiw__write1(stbi__write_context* s, unsigned char a) +{ + if ((size_t)s->buf_used + 1 > sizeof(s->buffer)) + stbiw__write_flush(s); + s->buffer[s->buf_used++] = a; +} + +static void stbiw__write3(stbi__write_context* s, unsigned char a, unsigned char b, unsigned char c) +{ + int n; + if ((size_t)s->buf_used + 3 > sizeof(s->buffer)) + stbiw__write_flush(s); + n = s->buf_used; + s->buf_used = n + 3; + s->buffer[n + 0] = a; + s->buffer[n + 1] = b; + s->buffer[n + 2] = c; +} + +static void stbiw__write_pixel(stbi__write_context* s, int rgb_dir, int comp, int write_alpha, int expand_mono, unsigned char* d) +{ + unsigned char bg[3] = { 255, 0, 255 }, px[3]; + int k; + + if (write_alpha < 0) + stbiw__write1(s, d[comp - 1]); + + switch (comp) { + case 2: // 2 pixels = mono + alpha, alpha is written separately, so same as 1-channel case + case 1: + if (expand_mono) + stbiw__write3(s, d[0], d[0], d[0]); // monochrome bmp + else + stbiw__write1(s, d[0]); // monochrome TGA + break; + case 4: + if (!write_alpha) { + // composite against pink background + for (k = 0; k < 3; ++k) + px[k] = bg[k] + ((d[k] - bg[k]) * d[3]) / 255; + stbiw__write3(s, px[1 - rgb_dir], px[1], px[1 + rgb_dir]); + break; + } + /* FALLTHROUGH */ + case 3: + stbiw__write3(s, d[1 - rgb_dir], d[1], d[1 + rgb_dir]); + break; + } + if (write_alpha > 0) + stbiw__write1(s, d[comp - 1]); +} + +static void stbiw__write_pixels(stbi__write_context* s, int rgb_dir, int vdir, int x, int y, int comp, void* data, int write_alpha, int scanline_pad, int expand_mono) +{ + stbiw_uint32 zero = 0; + int i, j, j_end; + + if (y <= 0) + return; + + if (stbi__flip_vertically_on_write) + vdir *= -1; + + if (vdir < 0) { + j_end = -1; + j = y - 1; + } else { + j_end = y; + j = 0; + } + + for (; j != j_end; j += vdir) { + for (i = 0; i < x; ++i) { + unsigned char* d = (unsigned char*)data + (j * x + i) * comp; + stbiw__write_pixel(s, rgb_dir, comp, write_alpha, expand_mono, d); + } + stbiw__write_flush(s); + s->func(s->context, &zero, scanline_pad); + } +} + +static int stbiw__outfile(stbi__write_context* s, int rgb_dir, int vdir, int x, int y, int comp, int expand_mono, void* data, int alpha, int pad, char const* fmt, ...) +{ + if (y < 0 || x < 0) { + return 0; + } else { + va_list v; + va_start(v, fmt); + stbiw__writefv(s, fmt, v); + va_end(v); + stbiw__write_pixels(s, rgb_dir, vdir, x, y, comp, data, alpha, pad, expand_mono); + return 1; + } +} + +static int stbi_write_bmp_core(stbi__write_context* s, int x, int y, int comp, void const* data) +{ + if (comp != 4) { + // write RGB bitmap + int pad = (-x * 3) & 3; + return stbiw__outfile(s, -1, -1, x, y, comp, 1, (void*)data, 0, pad, + "11 4 22 4" + "4 44 22 444444", + 'B', 'M', 14 + 40 + (x * 3 + pad) * y, 0, 0, 14 + 40, // file header + 40, x, y, 1, 24, 0, 0, 0, 0, 0, 0); // bitmap header + } else { + // RGBA bitmaps need a v4 header + // use BI_BITFIELDS mode with 32bpp and alpha mask + // (straight BI_RGB with alpha mask doesn't work in most readers) + return stbiw__outfile(s, -1, -1, x, y, comp, 1, (void*)data, 1, 0, + "11 4 22 4" + "4 44 22 444444 4444 4 444 444 444 444", + 'B', 'M', 14 + 108 + x * y * 4, 0, 0, 14 + 108, // file header + 108, x, y, 1, 32, 3, 0, 0, 0, 0, 0, 0xff0000, 0xff00, 0xff, 0xff000000u, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0); // bitmap V4 header + } +} + +STBIWDEF int stbi_write_bmp_to_func(stbi_write_func* func, void* context, int x, int y, int comp, void const* data) +{ + stbi__write_context s = { 0 }; + stbi__start_write_callbacks(&s, func, context); + return stbi_write_bmp_core(&s, x, y, comp, data); +} + +# ifndef STBI_WRITE_NO_STDIO +STBIWDEF int stbi_write_bmp(char const* filename, int x, int y, int comp, void const* data) +{ + stbi__write_context s = { 0 }; + if (stbi__start_write_file(&s, filename)) { + int r = stbi_write_bmp_core(&s, x, y, comp, data); + stbi__end_write_file(&s); + return r; + } else + return 0; +} +# endif //! STBI_WRITE_NO_STDIO + +static int stbi_write_tga_core(stbi__write_context* s, int x, int y, int comp, void* data) +{ + int has_alpha = (comp == 2 || comp == 4); + int colorbytes = has_alpha ? comp - 1 : comp; + int format = colorbytes < 2 ? 3 : 2; // 3 color channels (RGB/RGBA) = 2, 1 color channel (Y/YA) = 3 + + if (y < 0 || x < 0) + return 0; + + if (!stbi_write_tga_with_rle) { + return stbiw__outfile(s, -1, -1, x, y, comp, 0, (void*)data, has_alpha, 0, + "111 221 2222 11", 0, 0, format, 0, 0, 0, 0, 0, x, y, (colorbytes + has_alpha) * 8, has_alpha * 8); + } else { + int i, j, k; + int jend, jdir; + + stbiw__writef(s, "111 221 2222 11", 0, 0, format + 8, 0, 0, 0, 0, 0, x, y, (colorbytes + has_alpha) * 8, has_alpha * 8); + + if (stbi__flip_vertically_on_write) { + j = 0; + jend = y; + jdir = 1; + } else { + j = y - 1; + jend = -1; + jdir = -1; + } + for (; j != jend; j += jdir) { + unsigned char* row = (unsigned char*)data + j * x * comp; + int len; + + for (i = 0; i < x; i += len) { + unsigned char* begin = row + i * comp; + int diff = 1; + len = 1; + + if (i < x - 1) { + ++len; + diff = memcmp(begin, row + (i + 1) * comp, comp); + if (diff) { + unsigned char const* prev = begin; + for (k = i + 2; k < x && len < 128; ++k) { + if (memcmp(prev, row + k * comp, comp)) { + prev += comp; + ++len; + } else { + --len; + break; + } + } + } else { + for (k = i + 2; k < x && len < 128; ++k) { + if (!memcmp(begin, row + k * comp, comp)) { + ++len; + } else { + break; + } + } + } + } + + if (diff) { + unsigned char header = STBIW_UCHAR(len - 1); + stbiw__write1(s, header); + for (k = 0; k < len; ++k) { + stbiw__write_pixel(s, -1, comp, has_alpha, 0, begin + k * comp); + } + } else { + unsigned char header = STBIW_UCHAR(len - 129); + stbiw__write1(s, header); + stbiw__write_pixel(s, -1, comp, has_alpha, 0, begin); + } + } + } + stbiw__write_flush(s); + } + return 1; +} + +STBIWDEF int stbi_write_tga_to_func(stbi_write_func* func, void* context, int x, int y, int comp, void const* data) +{ + stbi__write_context s = { 0 }; + stbi__start_write_callbacks(&s, func, context); + return stbi_write_tga_core(&s, x, y, comp, (void*)data); +} + +# ifndef STBI_WRITE_NO_STDIO +STBIWDEF int stbi_write_tga(char const* filename, int x, int y, int comp, void const* data) +{ + stbi__write_context s = { 0 }; + if (stbi__start_write_file(&s, filename)) { + int r = stbi_write_tga_core(&s, x, y, comp, (void*)data); + stbi__end_write_file(&s); + return r; + } else + return 0; +} +# endif + +// ************************************************************************************************* +// Radiance RGBE HDR writer +// by Baldur Karlsson + +# define stbiw__max(a, b) ((a) > (b) ? (a) : (b)) + +# ifndef STBI_WRITE_NO_STDIO + +static void stbiw__linear_to_rgbe(unsigned char* rgbe, float* linear) +{ + int exponent; + float maxcomp = stbiw__max(linear[0], stbiw__max(linear[1], linear[2])); + + if (maxcomp < 1e-32f) { + rgbe[0] = rgbe[1] = rgbe[2] = rgbe[3] = 0; + } else { + float normalize = (float)frexp(maxcomp, &exponent) * 256.0f / maxcomp; + + rgbe[0] = (unsigned char)(linear[0] * normalize); + rgbe[1] = (unsigned char)(linear[1] * normalize); + rgbe[2] = (unsigned char)(linear[2] * normalize); + rgbe[3] = (unsigned char)(exponent + 128); + } +} + +static void stbiw__write_run_data(stbi__write_context* s, int length, unsigned char databyte) +{ + unsigned char lengthbyte = STBIW_UCHAR(length + 128); + STBIW_ASSERT(length + 128 <= 255); + s->func(s->context, &lengthbyte, 1); + s->func(s->context, &databyte, 1); +} + +static void stbiw__write_dump_data(stbi__write_context* s, int length, unsigned char* data) +{ + unsigned char lengthbyte = STBIW_UCHAR(length); + STBIW_ASSERT(length <= 128); // inconsistent with spec but consistent with official code + s->func(s->context, &lengthbyte, 1); + s->func(s->context, data, length); +} + +static void stbiw__write_hdr_scanline(stbi__write_context* s, int width, int ncomp, unsigned char* scratch, float* scanline) +{ + unsigned char scanlineheader[4] = { 2, 2, 0, 0 }; + unsigned char rgbe[4]; + float linear[3]; + int x; + + scanlineheader[2] = (width & 0xff00) >> 8; + scanlineheader[3] = (width & 0x00ff); + + /* skip RLE for images too small or large */ + if (width < 8 || width >= 32768) { + for (x = 0; x < width; x++) { + switch (ncomp) { + case 4: /* fallthrough */ + case 3: + linear[2] = scanline[x * ncomp + 2]; + linear[1] = scanline[x * ncomp + 1]; + linear[0] = scanline[x * ncomp + 0]; + break; + default: + linear[0] = linear[1] = linear[2] = scanline[x * ncomp + 0]; + break; + } + stbiw__linear_to_rgbe(rgbe, linear); + s->func(s->context, rgbe, 4); + } + } else { + int c, r; + /* encode into scratch buffer */ + for (x = 0; x < width; x++) { + switch (ncomp) { + case 4: /* fallthrough */ + case 3: + linear[2] = scanline[x * ncomp + 2]; + linear[1] = scanline[x * ncomp + 1]; + linear[0] = scanline[x * ncomp + 0]; + break; + default: + linear[0] = linear[1] = linear[2] = scanline[x * ncomp + 0]; + break; + } + stbiw__linear_to_rgbe(rgbe, linear); + scratch[x + width * 0] = rgbe[0]; + scratch[x + width * 1] = rgbe[1]; + scratch[x + width * 2] = rgbe[2]; + scratch[x + width * 3] = rgbe[3]; + } + + s->func(s->context, scanlineheader, 4); + + /* RLE each component separately */ + for (c = 0; c < 4; c++) { + unsigned char* comp = &scratch[width * c]; + + x = 0; + while (x < width) { + // find first run + r = x; + while (r + 2 < width) { + if (comp[r] == comp[r + 1] && comp[r] == comp[r + 2]) + break; + ++r; + } + if (r + 2 >= width) + r = width; + // dump up to first run + while (x < r) { + int len = r - x; + if (len > 128) + len = 128; + stbiw__write_dump_data(s, len, &comp[x]); + x += len; + } + // if there's a run, output it + if (r + 2 < width) { // same test as what we break out of in search loop, so only true if we break'd + // find next byte after run + while (r < width && comp[r] == comp[x]) + ++r; + // output run up to r + while (x < r) { + int len = r - x; + if (len > 127) + len = 127; + stbiw__write_run_data(s, len, comp[x]); + x += len; + } + } + } + } + } +} + +static int stbi_write_hdr_core(stbi__write_context* s, int x, int y, int comp, float* data) +{ + if (y <= 0 || x <= 0 || data == NULL) + return 0; + else { + // Each component is stored separately. Allocate scratch space for full output scanline. + unsigned char* scratch = (unsigned char*)STBIW_MALLOC(x * 4); + int i, len; + char buffer[128]; + char header[] = "#?RADIANCE\n# Written by stb_image_write.h\nFORMAT=32-bit_rle_rgbe\n"; + s->func(s->context, header, sizeof(header) - 1); + +# ifdef __STDC_LIB_EXT1__ + len = sprintf_s(buffer, sizeof(buffer), "EXPOSURE= 1.0000000000000\n\n-Y %d +X %d\n", y, x); +# else + len = sprintf(buffer, "EXPOSURE= 1.0000000000000\n\n-Y %d +X %d\n", y, x); +# endif + s->func(s->context, buffer, len); + + for (i = 0; i < y; i++) + stbiw__write_hdr_scanline(s, x, comp, scratch, data + comp * x * (stbi__flip_vertically_on_write ? y - 1 - i : i)); + STBIW_FREE(scratch); + return 1; + } +} + +STBIWDEF int stbi_write_hdr_to_func(stbi_write_func* func, void* context, int x, int y, int comp, float const* data) +{ + stbi__write_context s = { 0 }; + stbi__start_write_callbacks(&s, func, context); + return stbi_write_hdr_core(&s, x, y, comp, (float*)data); +} + +STBIWDEF int stbi_write_hdr(char const* filename, int x, int y, int comp, float const* data) +{ + stbi__write_context s = { 0 }; + if (stbi__start_write_file(&s, filename)) { + int r = stbi_write_hdr_core(&s, x, y, comp, (float*)data); + stbi__end_write_file(&s); + return r; + } else + return 0; +} +# endif // STBI_WRITE_NO_STDIO + +////////////////////////////////////////////////////////////////////////////// +// +// PNG writer +// + +# ifndef STBIW_ZLIB_COMPRESS +// stretchy buffer; stbiw__sbpush() == vector<>::push_back() -- stbiw__sbcount() == vector<>::size() +# define stbiw__sbraw(a) ((int*)(void*)(a) - 2) +# define stbiw__sbm(a) stbiw__sbraw(a)[0] +# define stbiw__sbn(a) stbiw__sbraw(a)[1] + +# define stbiw__sbneedgrow(a, n) ((a) == 0 || stbiw__sbn(a) + n >= stbiw__sbm(a)) +# define stbiw__sbmaybegrow(a, n) (stbiw__sbneedgrow(a, (n)) ? stbiw__sbgrow(a, n) : 0) +# define stbiw__sbgrow(a, n) stbiw__sbgrowf((void**)&(a), (n), sizeof(*(a))) + +# define stbiw__sbpush(a, v) (stbiw__sbmaybegrow(a, 1), (a)[stbiw__sbn(a)++] = (v)) +# define stbiw__sbcount(a) ((a) ? stbiw__sbn(a) : 0) +# define stbiw__sbfree(a) ((a) ? STBIW_FREE(stbiw__sbraw(a)), 0 : 0) + +static void* stbiw__sbgrowf(void** arr, int increment, int itemsize) +{ + int m = *arr ? 2 * stbiw__sbm(*arr) + increment : increment + 1; + void* p = STBIW_REALLOC_SIZED(*arr ? stbiw__sbraw(*arr) : 0, *arr ? (stbiw__sbm(*arr) * itemsize + sizeof(int) * 2) : 0, itemsize * m + sizeof(int) * 2); + STBIW_ASSERT(p); + if (p) { + if (!*arr) + ((int*)p)[1] = 0; + *arr = (void*)((int*)p + 2); + stbiw__sbm(*arr) = m; + } + return *arr; +} + +static unsigned char* stbiw__zlib_flushf(unsigned char* data, unsigned int* bitbuffer, int* bitcount) +{ + while (*bitcount >= 8) { + stbiw__sbpush(data, STBIW_UCHAR(*bitbuffer)); + *bitbuffer >>= 8; + *bitcount -= 8; + } + return data; +} + +static int stbiw__zlib_bitrev(int code, int codebits) +{ + int res = 0; + while (codebits--) { + res = (res << 1) | (code & 1); + code >>= 1; + } + return res; +} + +static unsigned int stbiw__zlib_countm(unsigned char* a, unsigned char* b, int limit) +{ + int i; + for (i = 0; i < limit && i < 258; ++i) + if (a[i] != b[i]) + break; + return i; +} + +static unsigned int stbiw__zhash(unsigned char* data) +{ + stbiw_uint32 hash = data[0] + (data[1] << 8) + (data[2] << 16); + hash ^= hash << 3; + hash += hash >> 5; + hash ^= hash << 4; + hash += hash >> 17; + hash ^= hash << 25; + hash += hash >> 6; + return hash; +} + +# define stbiw__zlib_flush() (out = stbiw__zlib_flushf(out, &bitbuf, &bitcount)) +# define stbiw__zlib_add(code, codebits) \ + (bitbuf |= (code) << bitcount, bitcount += (codebits), stbiw__zlib_flush()) +# define stbiw__zlib_huffa(b, c) stbiw__zlib_add(stbiw__zlib_bitrev(b, c), c) +// default huffman tables +# define stbiw__zlib_huff1(n) stbiw__zlib_huffa(0x30 + (n), 8) +# define stbiw__zlib_huff2(n) stbiw__zlib_huffa(0x190 + (n) - 144, 9) +# define stbiw__zlib_huff3(n) stbiw__zlib_huffa(0 + (n) - 256, 7) +# define stbiw__zlib_huff4(n) stbiw__zlib_huffa(0xc0 + (n) - 280, 8) +# define stbiw__zlib_huff(n) ((n) <= 143 ? stbiw__zlib_huff1(n) : (n) <= 255 ? stbiw__zlib_huff2(n) \ + : (n) <= 279 ? stbiw__zlib_huff3(n) \ + : stbiw__zlib_huff4(n)) +# define stbiw__zlib_huffb(n) ((n) <= 143 ? stbiw__zlib_huff1(n) : stbiw__zlib_huff2(n)) + +# define stbiw__ZHASH 16384 + +# endif // STBIW_ZLIB_COMPRESS + +STBIWDEF unsigned char* stbi_zlib_compress(unsigned char* data, int data_len, int* out_len, int quality) +{ +# ifdef STBIW_ZLIB_COMPRESS + // user provided a zlib compress implementation, use that + return STBIW_ZLIB_COMPRESS(data, data_len, out_len, quality); +# else // use builtin + static unsigned short lengthc[] = { 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31, 35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 259 }; + static unsigned char lengtheb[] = { 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0 }; + static unsigned short distc[] = { 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193, 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145, 8193, 12289, 16385, 24577, 32768 }; + static unsigned char disteb[] = { 0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13 }; + unsigned int bitbuf = 0; + int i, j, bitcount = 0; + unsigned char* out = NULL; + unsigned char*** hash_table = (unsigned char***)STBIW_MALLOC(stbiw__ZHASH * sizeof(unsigned char**)); + if (hash_table == NULL) + return NULL; + if (quality < 5) + quality = 5; + + stbiw__sbpush(out, 0x78); // DEFLATE 32K window + stbiw__sbpush(out, 0x5e); // FLEVEL = 1 + stbiw__zlib_add(1, 1); // BFINAL = 1 + stbiw__zlib_add(1, 2); // BTYPE = 1 -- fixed huffman + + for (i = 0; i < stbiw__ZHASH; ++i) + hash_table[i] = NULL; + + i = 0; + while (i < data_len - 3) { + // hash next 3 bytes of data to be compressed + int h = stbiw__zhash(data + i) & (stbiw__ZHASH - 1), best = 3; + unsigned char* bestloc = 0; + unsigned char** hlist = hash_table[h]; + int n = stbiw__sbcount(hlist); + for (j = 0; j < n; ++j) { + if (hlist[j] - data > i - 32768) { // if entry lies within window + int d = stbiw__zlib_countm(hlist[j], data + i, data_len - i); + if (d >= best) { + best = d; + bestloc = hlist[j]; + } + } + } + // when hash table entry is too long, delete half the entries + if (hash_table[h] && stbiw__sbn(hash_table[h]) == 2 * quality) { + STBIW_MEMMOVE(hash_table[h], hash_table[h] + quality, sizeof(hash_table[h][0]) * quality); + stbiw__sbn(hash_table[h]) = quality; + } + stbiw__sbpush(hash_table[h], data + i); + + if (bestloc) { + // "lazy matching" - check match at *next* byte, and if it's better, do cur byte as literal + h = stbiw__zhash(data + i + 1) & (stbiw__ZHASH - 1); + hlist = hash_table[h]; + n = stbiw__sbcount(hlist); + for (j = 0; j < n; ++j) { + if (hlist[j] - data > i - 32767) { + int e = stbiw__zlib_countm(hlist[j], data + i + 1, data_len - i - 1); + if (e > best) { // if next match is better, bail on current match + bestloc = NULL; + break; + } + } + } + } + + if (bestloc) { + int d = (int)(data + i - bestloc); // distance back + STBIW_ASSERT(d <= 32767 && best <= 258); + for (j = 0; best > lengthc[j + 1] - 1; ++j) + ; + stbiw__zlib_huff(j + 257); + if (lengtheb[j]) + stbiw__zlib_add(best - lengthc[j], lengtheb[j]); + for (j = 0; d > distc[j + 1] - 1; ++j) + ; + stbiw__zlib_add(stbiw__zlib_bitrev(j, 5), 5); + if (disteb[j]) + stbiw__zlib_add(d - distc[j], disteb[j]); + i += best; + } else { + stbiw__zlib_huffb(data[i]); + ++i; + } + } + // write out final bytes + for (; i < data_len; ++i) + stbiw__zlib_huffb(data[i]); + stbiw__zlib_huff(256); // end of block + // pad with 0 bits to byte boundary + while (bitcount) + stbiw__zlib_add(0, 1); + + for (i = 0; i < stbiw__ZHASH; ++i) + (void)stbiw__sbfree(hash_table[i]); + STBIW_FREE(hash_table); + + // store uncompressed instead if compression was worse + if (stbiw__sbn(out) > data_len + 2 + ((data_len + 32766) / 32767) * 5) { + stbiw__sbn(out) = 2; // truncate to DEFLATE 32K window and FLEVEL = 1 + for (j = 0; j < data_len;) { + int blocklen = data_len - j; + if (blocklen > 32767) + blocklen = 32767; + stbiw__sbpush(out, data_len - j == blocklen); // BFINAL = ?, BTYPE = 0 -- no compression + stbiw__sbpush(out, STBIW_UCHAR(blocklen)); // LEN + stbiw__sbpush(out, STBIW_UCHAR(blocklen >> 8)); + stbiw__sbpush(out, STBIW_UCHAR(~blocklen)); // NLEN + stbiw__sbpush(out, STBIW_UCHAR(~blocklen >> 8)); + memcpy(out + stbiw__sbn(out), data + j, blocklen); + stbiw__sbn(out) += blocklen; + j += blocklen; + } + } + + { + // compute adler32 on input + unsigned int s1 = 1, s2 = 0; + int blocklen = (int)(data_len % 5552); + j = 0; + while (j < data_len) { + for (i = 0; i < blocklen; ++i) { + s1 += data[j + i]; + s2 += s1; + } + s1 %= 65521; + s2 %= 65521; + j += blocklen; + blocklen = 5552; + } + stbiw__sbpush(out, STBIW_UCHAR(s2 >> 8)); + stbiw__sbpush(out, STBIW_UCHAR(s2)); + stbiw__sbpush(out, STBIW_UCHAR(s1 >> 8)); + stbiw__sbpush(out, STBIW_UCHAR(s1)); + } + *out_len = stbiw__sbn(out); + // make returned pointer freeable + STBIW_MEMMOVE(stbiw__sbraw(out), out, *out_len); + return (unsigned char*)stbiw__sbraw(out); +# endif // STBIW_ZLIB_COMPRESS +} + +static unsigned int stbiw__crc32(unsigned char* buffer, int len) +{ +# ifdef STBIW_CRC32 + return STBIW_CRC32(buffer, len); +# else + static unsigned int crc_table[256] = { + 0x00000000, 0x77073096, 0xEE0E612C, 0x990951BA, 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3, + 0x0eDB8832, 0x79DCB8A4, 0xE0D5E91E, 0x97D2D988, 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91, + 0x1DB71064, 0x6AB020F2, 0xF3B97148, 0x84BE41DE, 0x1ADAD47D, 0x6DDDE4EB, 0xF4D4B551, 0x83D385C7, + 0x136C9856, 0x646BA8C0, 0xFD62F97A, 0x8A65C9EC, 0x14015C4F, 0x63066CD9, 0xFA0F3D63, 0x8D080DF5, + 0x3B6E20C8, 0x4C69105E, 0xD56041E4, 0xA2677172, 0x3C03E4D1, 0x4B04D447, 0xD20D85FD, 0xA50AB56B, + 0x35B5A8FA, 0x42B2986C, 0xDBBBC9D6, 0xACBCF940, 0x32D86CE3, 0x45DF5C75, 0xDCD60DCF, 0xABD13D59, + 0x26D930AC, 0x51DE003A, 0xC8D75180, 0xBFD06116, 0x21B4F4B5, 0x56B3C423, 0xCFBA9599, 0xB8BDA50F, + 0x2802B89E, 0x5F058808, 0xC60CD9B2, 0xB10BE924, 0x2F6F7C87, 0x58684C11, 0xC1611DAB, 0xB6662D3D, + 0x76DC4190, 0x01DB7106, 0x98D220BC, 0xEFD5102A, 0x71B18589, 0x06B6B51F, 0x9FBFE4A5, 0xE8B8D433, + 0x7807C9A2, 0x0F00F934, 0x9609A88E, 0xE10E9818, 0x7F6A0DBB, 0x086D3D2D, 0x91646C97, 0xE6635C01, + 0x6B6B51F4, 0x1C6C6162, 0x856530D8, 0xF262004E, 0x6C0695ED, 0x1B01A57B, 0x8208F4C1, 0xF50FC457, + 0x65B0D9C6, 0x12B7E950, 0x8BBEB8EA, 0xFCB9887C, 0x62DD1DDF, 0x15DA2D49, 0x8CD37CF3, 0xFBD44C65, + 0x4DB26158, 0x3AB551CE, 0xA3BC0074, 0xD4BB30E2, 0x4ADFA541, 0x3DD895D7, 0xA4D1C46D, 0xD3D6F4FB, + 0x4369E96A, 0x346ED9FC, 0xAD678846, 0xDA60B8D0, 0x44042D73, 0x33031DE5, 0xAA0A4C5F, 0xDD0D7CC9, + 0x5005713C, 0x270241AA, 0xBE0B1010, 0xC90C2086, 0x5768B525, 0x206F85B3, 0xB966D409, 0xCE61E49F, + 0x5EDEF90E, 0x29D9C998, 0xB0D09822, 0xC7D7A8B4, 0x59B33D17, 0x2EB40D81, 0xB7BD5C3B, 0xC0BA6CAD, + 0xEDB88320, 0x9ABFB3B6, 0x03B6E20C, 0x74B1D29A, 0xEAD54739, 0x9DD277AF, 0x04DB2615, 0x73DC1683, + 0xE3630B12, 0x94643B84, 0x0D6D6A3E, 0x7A6A5AA8, 0xE40ECF0B, 0x9309FF9D, 0x0A00AE27, 0x7D079EB1, + 0xF00F9344, 0x8708A3D2, 0x1E01F268, 0x6906C2FE, 0xF762575D, 0x806567CB, 0x196C3671, 0x6E6B06E7, + 0xFED41B76, 0x89D32BE0, 0x10DA7A5A, 0x67DD4ACC, 0xF9B9DF6F, 0x8EBEEFF9, 0x17B7BE43, 0x60B08ED5, + 0xD6D6A3E8, 0xA1D1937E, 0x38D8C2C4, 0x4FDFF252, 0xD1BB67F1, 0xA6BC5767, 0x3FB506DD, 0x48B2364B, + 0xD80D2BDA, 0xAF0A1B4C, 0x36034AF6, 0x41047A60, 0xDF60EFC3, 0xA867DF55, 0x316E8EEF, 0x4669BE79, + 0xCB61B38C, 0xBC66831A, 0x256FD2A0, 0x5268E236, 0xCC0C7795, 0xBB0B4703, 0x220216B9, 0x5505262F, + 0xC5BA3BBE, 0xB2BD0B28, 0x2BB45A92, 0x5CB36A04, 0xC2D7FFA7, 0xB5D0CF31, 0x2CD99E8B, 0x5BDEAE1D, + 0x9B64C2B0, 0xEC63F226, 0x756AA39C, 0x026D930A, 0x9C0906A9, 0xEB0E363F, 0x72076785, 0x05005713, + 0x95BF4A82, 0xE2B87A14, 0x7BB12BAE, 0x0CB61B38, 0x92D28E9B, 0xE5D5BE0D, 0x7CDCEFB7, 0x0BDBDF21, + 0x86D3D2D4, 0xF1D4E242, 0x68DDB3F8, 0x1FDA836E, 0x81BE16CD, 0xF6B9265B, 0x6FB077E1, 0x18B74777, + 0x88085AE6, 0xFF0F6A70, 0x66063BCA, 0x11010B5C, 0x8F659EFF, 0xF862AE69, 0x616BFFD3, 0x166CCF45, + 0xA00AE278, 0xD70DD2EE, 0x4E048354, 0x3903B3C2, 0xA7672661, 0xD06016F7, 0x4969474D, 0x3E6E77DB, + 0xAED16A4A, 0xD9D65ADC, 0x40DF0B66, 0x37D83BF0, 0xA9BCAE53, 0xDEBB9EC5, 0x47B2CF7F, 0x30B5FFE9, + 0xBDBDF21C, 0xCABAC28A, 0x53B39330, 0x24B4A3A6, 0xBAD03605, 0xCDD70693, 0x54DE5729, 0x23D967BF, + 0xB3667A2E, 0xC4614AB8, 0x5D681B02, 0x2A6F2B94, 0xB40BBE37, 0xC30C8EA1, 0x5A05DF1B, 0x2D02EF8D + }; + + unsigned int crc = ~0u; + int i; + for (i = 0; i < len; ++i) + crc = (crc >> 8) ^ crc_table[buffer[i] ^ (crc & 0xff)]; + return ~crc; +# endif +} + +# define stbiw__wpng4(o, a, b, c, d) ((o)[0] = STBIW_UCHAR(a), (o)[1] = STBIW_UCHAR(b), (o)[2] = STBIW_UCHAR(c), (o)[3] = STBIW_UCHAR(d), (o) += 4) +# define stbiw__wp32(data, v) stbiw__wpng4(data, (v) >> 24, (v) >> 16, (v) >> 8, (v)); +# define stbiw__wptag(data, s) stbiw__wpng4(data, s[0], s[1], s[2], s[3]) + +static void stbiw__wpcrc(unsigned char** data, int len) +{ + unsigned int crc = stbiw__crc32(*data - len - 4, len + 4); + stbiw__wp32(*data, crc); +} + +static unsigned char stbiw__paeth(int a, int b, int c) +{ + int p = a + b - c, pa = abs(p - a), pb = abs(p - b), pc = abs(p - c); + if (pa <= pb && pa <= pc) + return STBIW_UCHAR(a); + if (pb <= pc) + return STBIW_UCHAR(b); + return STBIW_UCHAR(c); +} + +// @OPTIMIZE: provide an option that always forces left-predict or paeth predict +static void stbiw__encode_png_line(unsigned char* pixels, int stride_bytes, int width, int height, int y, int n, int filter_type, signed char* line_buffer) +{ + static int mapping[] = { 0, 1, 2, 3, 4 }; + static int firstmap[] = { 0, 1, 0, 5, 6 }; + int* mymap = (y != 0) ? mapping : firstmap; + int i; + int type = mymap[filter_type]; + unsigned char* z = pixels + stride_bytes * (stbi__flip_vertically_on_write ? height - 1 - y : y); + int signed_stride = stbi__flip_vertically_on_write ? -stride_bytes : stride_bytes; + + if (type == 0) { + memcpy(line_buffer, z, width * n); + return; + } + + // first loop isn't optimized since it's just one pixel + for (i = 0; i < n; ++i) { + switch (type) { + case 1: + line_buffer[i] = z[i]; + break; + case 2: + line_buffer[i] = z[i] - z[i - signed_stride]; + break; + case 3: + line_buffer[i] = z[i] - (z[i - signed_stride] >> 1); + break; + case 4: + line_buffer[i] = (signed char)(z[i] - stbiw__paeth(0, z[i - signed_stride], 0)); + break; + case 5: + line_buffer[i] = z[i]; + break; + case 6: + line_buffer[i] = z[i]; + break; + } + } + switch (type) { + case 1: + for (i = n; i < width * n; ++i) + line_buffer[i] = z[i] - z[i - n]; + break; + case 2: + for (i = n; i < width * n; ++i) + line_buffer[i] = z[i] - z[i - signed_stride]; + break; + case 3: + for (i = n; i < width * n; ++i) + line_buffer[i] = z[i] - ((z[i - n] + z[i - signed_stride]) >> 1); + break; + case 4: + for (i = n; i < width * n; ++i) + line_buffer[i] = z[i] - stbiw__paeth(z[i - n], z[i - signed_stride], z[i - signed_stride - n]); + break; + case 5: + for (i = n; i < width * n; ++i) + line_buffer[i] = z[i] - (z[i - n] >> 1); + break; + case 6: + for (i = n; i < width * n; ++i) + line_buffer[i] = z[i] - stbiw__paeth(z[i - n], 0, 0); + break; + } +} + +STBIWDEF unsigned char* stbi_write_png_to_mem(unsigned char const* pixels, int stride_bytes, int x, int y, int n, int* out_len) +{ + int force_filter = stbi_write_force_png_filter; + int ctype[5] = { -1, 0, 4, 2, 6 }; + unsigned char sig[8] = { 137, 80, 78, 71, 13, 10, 26, 10 }; + unsigned char *out, *o, *filt, *zlib; + signed char* line_buffer; + int j, zlen; + + if (stride_bytes == 0) + stride_bytes = x * n; + + if (force_filter >= 5) { + force_filter = -1; + } + + filt = (unsigned char*)STBIW_MALLOC((x * n + 1) * y); + if (!filt) + return 0; + line_buffer = (signed char*)STBIW_MALLOC(x * n); + if (!line_buffer) { + STBIW_FREE(filt); + return 0; + } + for (j = 0; j < y; ++j) { + int filter_type; + if (force_filter > -1) { + filter_type = force_filter; + stbiw__encode_png_line((unsigned char*)(pixels), stride_bytes, x, y, j, n, force_filter, line_buffer); + } else { // Estimate the best filter by running through all of them: + int best_filter = 0, best_filter_val = 0x7fffffff, est, i; + for (filter_type = 0; filter_type < 5; filter_type++) { + stbiw__encode_png_line((unsigned char*)(pixels), stride_bytes, x, y, j, n, filter_type, line_buffer); + + // Estimate the entropy of the line using this filter; the less, the better. + est = 0; + for (i = 0; i < x * n; ++i) { + est += abs((signed char)line_buffer[i]); + } + if (est < best_filter_val) { + best_filter_val = est; + best_filter = filter_type; + } + } + if (filter_type != best_filter) { // If the last iteration already got us the best filter, don't redo it + stbiw__encode_png_line((unsigned char*)(pixels), stride_bytes, x, y, j, n, best_filter, line_buffer); + filter_type = best_filter; + } + } + // when we get here, filter_type contains the filter type, and line_buffer contains the data + filt[j * (x * n + 1)] = (unsigned char)filter_type; + STBIW_MEMMOVE(filt + j * (x * n + 1) + 1, line_buffer, x * n); + } + STBIW_FREE(line_buffer); + zlib = stbi_zlib_compress(filt, y * (x * n + 1), &zlen, stbi_write_png_compression_level); + STBIW_FREE(filt); + if (!zlib) + return 0; + + // each tag requires 12 bytes of overhead + out = (unsigned char*)STBIW_MALLOC(8 + 12 + 13 + 12 + zlen + 12); + if (!out) + return 0; + *out_len = 8 + 12 + 13 + 12 + zlen + 12; + + o = out; + STBIW_MEMMOVE(o, sig, 8); + o += 8; + stbiw__wp32(o, 13); // header length + stbiw__wptag(o, "IHDR"); + stbiw__wp32(o, x); + stbiw__wp32(o, y); + *o++ = 8; + *o++ = STBIW_UCHAR(ctype[n]); + *o++ = 0; + *o++ = 0; + *o++ = 0; + stbiw__wpcrc(&o, 13); + + stbiw__wp32(o, zlen); + stbiw__wptag(o, "IDAT"); + STBIW_MEMMOVE(o, zlib, zlen); + o += zlen; + STBIW_FREE(zlib); + stbiw__wpcrc(&o, zlen); + + stbiw__wp32(o, 0); + stbiw__wptag(o, "IEND"); + stbiw__wpcrc(&o, 0); + + STBIW_ASSERT(o == out + *out_len); + + return out; +} + +# ifndef STBI_WRITE_NO_STDIO +STBIWDEF int stbi_write_png(char const* filename, int x, int y, int comp, void const* data, int stride_bytes) +{ + FILE* f; + int len; + unsigned char* png = stbi_write_png_to_mem((unsigned char const*)data, stride_bytes, x, y, comp, &len); + if (png == NULL) + return 0; + + f = stbiw__fopen(filename, "wb"); + if (!f) { + STBIW_FREE(png); + return 0; + } + fwrite(png, 1, len, f); + fclose(f); + STBIW_FREE(png); + return 1; +} +# endif + +STBIWDEF int stbi_write_png_to_func(stbi_write_func* func, void* context, int x, int y, int comp, void const* data, int stride_bytes) +{ + int len; + unsigned char* png = stbi_write_png_to_mem((unsigned char const*)data, stride_bytes, x, y, comp, &len); + if (png == NULL) + return 0; + func(context, png, len); + STBIW_FREE(png); + return 1; +} + +/* *************************************************************************** + * + * JPEG writer + * + * This is based on Jon Olick's jo_jpeg.cpp: + * public domain Simple, Minimalistic JPEG writer - http://www.jonolick.com/code.html + */ + +static unsigned char const stbiw__jpg_ZigZag[] = { 0, 1, 5, 6, 14, 15, 27, 28, 2, 4, 7, 13, 16, 26, 29, 42, 3, 8, 12, 17, 25, 30, 41, 43, 9, 11, 18, + 24, 31, 40, 44, 53, 10, 19, 23, 32, 39, 45, 52, 54, 20, 22, 33, 38, 46, 51, 55, 60, 21, 34, 37, 47, 50, 56, 59, 61, 35, 36, 48, 49, 57, 58, 62, 63 }; + +static void stbiw__jpg_writeBits(stbi__write_context* s, int* bitBufP, int* bitCntP, unsigned short const* bs) +{ + int bitBuf = *bitBufP, bitCnt = *bitCntP; + bitCnt += bs[1]; + bitBuf |= bs[0] << (24 - bitCnt); + while (bitCnt >= 8) { + unsigned char c = (bitBuf >> 16) & 255; + stbiw__putc(s, c); + if (c == 255) { + stbiw__putc(s, 0); + } + bitBuf <<= 8; + bitCnt -= 8; + } + *bitBufP = bitBuf; + *bitCntP = bitCnt; +} + +static void stbiw__jpg_DCT(float* d0p, float* d1p, float* d2p, float* d3p, float* d4p, float* d5p, float* d6p, float* d7p) +{ + float d0 = *d0p, d1 = *d1p, d2 = *d2p, d3 = *d3p, d4 = *d4p, d5 = *d5p, d6 = *d6p, d7 = *d7p; + float z1, z2, z3, z4, z5, z11, z13; + + float tmp0 = d0 + d7; + float tmp7 = d0 - d7; + float tmp1 = d1 + d6; + float tmp6 = d1 - d6; + float tmp2 = d2 + d5; + float tmp5 = d2 - d5; + float tmp3 = d3 + d4; + float tmp4 = d3 - d4; + + // Even part + float tmp10 = tmp0 + tmp3; // phase 2 + float tmp13 = tmp0 - tmp3; + float tmp11 = tmp1 + tmp2; + float tmp12 = tmp1 - tmp2; + + d0 = tmp10 + tmp11; // phase 3 + d4 = tmp10 - tmp11; + + z1 = (tmp12 + tmp13) * 0.707106781f; // c4 + d2 = tmp13 + z1; // phase 5 + d6 = tmp13 - z1; + + // Odd part + tmp10 = tmp4 + tmp5; // phase 2 + tmp11 = tmp5 + tmp6; + tmp12 = tmp6 + tmp7; + + // The rotator is modified from fig 4-8 to avoid extra negations. + z5 = (tmp10 - tmp12) * 0.382683433f; // c6 + z2 = tmp10 * 0.541196100f + z5; // c2-c6 + z4 = tmp12 * 1.306562965f + z5; // c2+c6 + z3 = tmp11 * 0.707106781f; // c4 + + z11 = tmp7 + z3; // phase 5 + z13 = tmp7 - z3; + + *d5p = z13 + z2; // phase 6 + *d3p = z13 - z2; + *d1p = z11 + z4; + *d7p = z11 - z4; + + *d0p = d0; + *d2p = d2; + *d4p = d4; + *d6p = d6; +} + +static void stbiw__jpg_calcBits(int val, unsigned short bits[2]) +{ + int tmp1 = val < 0 ? -val : val; + val = val < 0 ? val - 1 : val; + bits[1] = 1; + while (tmp1 >>= 1) { + ++bits[1]; + } + bits[0] = val & ((1 << bits[1]) - 1); +} + +static int stbiw__jpg_processDU(stbi__write_context* s, int* bitBuf, int* bitCnt, float* CDU, int du_stride, float* fdtbl, int DC, unsigned short const HTDC[256][2], unsigned short const HTAC[256][2]) +{ + unsigned short const EOB[2] = { HTAC[0x00][0], HTAC[0x00][1] }; + unsigned short const M16zeroes[2] = { HTAC[0xF0][0], HTAC[0xF0][1] }; + int dataOff, i, j, n, diff, end0pos, x, y; + int DU[64]; + + // DCT rows + for (dataOff = 0, n = du_stride * 8; dataOff < n; dataOff += du_stride) { + stbiw__jpg_DCT(&CDU[dataOff], &CDU[dataOff + 1], &CDU[dataOff + 2], &CDU[dataOff + 3], &CDU[dataOff + 4], &CDU[dataOff + 5], &CDU[dataOff + 6], &CDU[dataOff + 7]); + } + // DCT columns + for (dataOff = 0; dataOff < 8; ++dataOff) { + stbiw__jpg_DCT(&CDU[dataOff], &CDU[dataOff + du_stride], &CDU[dataOff + du_stride * 2], &CDU[dataOff + du_stride * 3], &CDU[dataOff + du_stride * 4], + &CDU[dataOff + du_stride * 5], &CDU[dataOff + du_stride * 6], &CDU[dataOff + du_stride * 7]); + } + // Quantize/descale/zigzag the coefficients + for (y = 0, j = 0; y < 8; ++y) { + for (x = 0; x < 8; ++x, ++j) { + float v; + i = y * du_stride + x; + v = CDU[i] * fdtbl[j]; + // DU[stbiw__jpg_ZigZag[j]] = (int)(v < 0 ? ceilf(v - 0.5f) : floorf(v + 0.5f)); + // ceilf() and floorf() are C99, not C89, but I /think/ they're not needed here anyway? + DU[stbiw__jpg_ZigZag[j]] = (int)(v < 0 ? v - 0.5f : v + 0.5f); + } + } + + // Encode DC + diff = DU[0] - DC; + if (diff == 0) { + stbiw__jpg_writeBits(s, bitBuf, bitCnt, HTDC[0]); + } else { + unsigned short bits[2]; + stbiw__jpg_calcBits(diff, bits); + stbiw__jpg_writeBits(s, bitBuf, bitCnt, HTDC[bits[1]]); + stbiw__jpg_writeBits(s, bitBuf, bitCnt, bits); + } + // Encode ACs + end0pos = 63; + for (; (end0pos > 0) && (DU[end0pos] == 0); --end0pos) { + } + // end0pos = first element in reverse order !=0 + if (end0pos == 0) { + stbiw__jpg_writeBits(s, bitBuf, bitCnt, EOB); + return DU[0]; + } + for (i = 1; i <= end0pos; ++i) { + int startpos = i; + int nrzeroes; + unsigned short bits[2]; + for (; DU[i] == 0 && i <= end0pos; ++i) { + } + nrzeroes = i - startpos; + if (nrzeroes >= 16) { + int lng = nrzeroes >> 4; + int nrmarker; + for (nrmarker = 1; nrmarker <= lng; ++nrmarker) + stbiw__jpg_writeBits(s, bitBuf, bitCnt, M16zeroes); + nrzeroes &= 15; + } + stbiw__jpg_calcBits(DU[i], bits); + stbiw__jpg_writeBits(s, bitBuf, bitCnt, HTAC[(nrzeroes << 4) + bits[1]]); + stbiw__jpg_writeBits(s, bitBuf, bitCnt, bits); + } + if (end0pos != 63) { + stbiw__jpg_writeBits(s, bitBuf, bitCnt, EOB); + } + return DU[0]; +} + +static int stbi_write_jpg_core(stbi__write_context* s, int width, int height, int comp, void const* data, int quality) +{ + // Constants that don't pollute global namespace + static unsigned char const std_dc_luminance_nrcodes[] = { 0, 0, 1, 5, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0 }; + static unsigned char const std_dc_luminance_values[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }; + static unsigned char const std_ac_luminance_nrcodes[] = { 0, 0, 2, 1, 3, 3, 2, 4, 3, 5, 5, 4, 4, 0, 0, 1, 0x7d }; + static unsigned char const std_ac_luminance_values[] = { + 0x01, 0x02, 0x03, 0x00, 0x04, 0x11, 0x05, 0x12, 0x21, 0x31, 0x41, 0x06, 0x13, 0x51, 0x61, 0x07, 0x22, 0x71, 0x14, 0x32, 0x81, 0x91, 0xa1, 0x08, + 0x23, 0x42, 0xb1, 0xc1, 0x15, 0x52, 0xd1, 0xf0, 0x24, 0x33, 0x62, 0x72, 0x82, 0x09, 0x0a, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x25, 0x26, 0x27, 0x28, + 0x29, 0x2a, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4a, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, + 0x5a, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6a, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7a, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, + 0x8a, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9a, 0xa2, 0xa3, 0xa4, 0xa5, 0xa6, 0xa7, 0xa8, 0xa9, 0xaa, 0xb2, 0xb3, 0xb4, 0xb5, 0xb6, + 0xb7, 0xb8, 0xb9, 0xba, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7, 0xd8, 0xd9, 0xda, 0xe1, 0xe2, + 0xe3, 0xe4, 0xe5, 0xe6, 0xe7, 0xe8, 0xe9, 0xea, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7, 0xf8, 0xf9, 0xfa + }; + static unsigned char const std_dc_chrominance_nrcodes[] = { 0, 0, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0 }; + static unsigned char const std_dc_chrominance_values[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }; + static unsigned char const std_ac_chrominance_nrcodes[] = { 0, 0, 2, 1, 2, 4, 4, 3, 4, 7, 5, 4, 4, 0, 1, 2, 0x77 }; + static unsigned char const std_ac_chrominance_values[] = { + 0x00, 0x01, 0x02, 0x03, 0x11, 0x04, 0x05, 0x21, 0x31, 0x06, 0x12, 0x41, 0x51, 0x07, 0x61, 0x71, 0x13, 0x22, 0x32, 0x81, 0x08, 0x14, 0x42, 0x91, + 0xa1, 0xb1, 0xc1, 0x09, 0x23, 0x33, 0x52, 0xf0, 0x15, 0x62, 0x72, 0xd1, 0x0a, 0x16, 0x24, 0x34, 0xe1, 0x25, 0xf1, 0x17, 0x18, 0x19, 0x1a, 0x26, + 0x27, 0x28, 0x29, 0x2a, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4a, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, + 0x59, 0x5a, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6a, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7a, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, + 0x88, 0x89, 0x8a, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9a, 0xa2, 0xa3, 0xa4, 0xa5, 0xa6, 0xa7, 0xa8, 0xa9, 0xaa, 0xb2, 0xb3, 0xb4, + 0xb5, 0xb6, 0xb7, 0xb8, 0xb9, 0xba, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7, 0xd8, 0xd9, 0xda, + 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7, 0xe8, 0xe9, 0xea, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7, 0xf8, 0xf9, 0xfa + }; + // Huffman tables + static unsigned short const YDC_HT[256][2] = { { 0, 2 }, { 2, 3 }, { 3, 3 }, { 4, 3 }, { 5, 3 }, { 6, 3 }, { 14, 4 }, { 30, 5 }, { 62, 6 }, { 126, 7 }, { 254, 8 }, { 510, 9 } }; + static unsigned short const UVDC_HT[256][2] = { { 0, 2 }, { 1, 2 }, { 2, 2 }, { 6, 3 }, { 14, 4 }, { 30, 5 }, { 62, 6 }, { 126, 7 }, { 254, 8 }, { 510, 9 }, { 1022, 10 }, { 2046, 11 } }; + static unsigned short const YAC_HT[256][2] = { + { 10, 4 }, { 0, 2 }, { 1, 2 }, { 4, 3 }, { 11, 4 }, { 26, 5 }, { 120, 7 }, { 248, 8 }, { 1014, 10 }, { 65410, 16 }, { 65411, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 12, 4 }, { 27, 5 }, { 121, 7 }, { 502, 9 }, { 2038, 11 }, { 65412, 16 }, { 65413, 16 }, { 65414, 16 }, { 65415, 16 }, { 65416, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 28, 5 }, { 249, 8 }, { 1015, 10 }, { 4084, 12 }, { 65417, 16 }, { 65418, 16 }, { 65419, 16 }, { 65420, 16 }, { 65421, 16 }, { 65422, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 58, 6 }, { 503, 9 }, { 4085, 12 }, { 65423, 16 }, { 65424, 16 }, { 65425, 16 }, { 65426, 16 }, { 65427, 16 }, { 65428, 16 }, { 65429, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 59, 6 }, { 1016, 10 }, { 65430, 16 }, { 65431, 16 }, { 65432, 16 }, { 65433, 16 }, { 65434, 16 }, { 65435, 16 }, { 65436, 16 }, { 65437, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 122, 7 }, { 2039, 11 }, { 65438, 16 }, { 65439, 16 }, { 65440, 16 }, { 65441, 16 }, { 65442, 16 }, { 65443, 16 }, { 65444, 16 }, { 65445, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 123, 7 }, { 4086, 12 }, { 65446, 16 }, { 65447, 16 }, { 65448, 16 }, { 65449, 16 }, { 65450, 16 }, { 65451, 16 }, { 65452, 16 }, { 65453, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 250, 8 }, { 4087, 12 }, { 65454, 16 }, { 65455, 16 }, { 65456, 16 }, { 65457, 16 }, { 65458, 16 }, { 65459, 16 }, { 65460, 16 }, { 65461, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 504, 9 }, { 32704, 15 }, { 65462, 16 }, { 65463, 16 }, { 65464, 16 }, { 65465, 16 }, { 65466, 16 }, { 65467, 16 }, { 65468, 16 }, { 65469, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 505, 9 }, { 65470, 16 }, { 65471, 16 }, { 65472, 16 }, { 65473, 16 }, { 65474, 16 }, { 65475, 16 }, { 65476, 16 }, { 65477, 16 }, { 65478, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 506, 9 }, { 65479, 16 }, { 65480, 16 }, { 65481, 16 }, { 65482, 16 }, { 65483, 16 }, { 65484, 16 }, { 65485, 16 }, { 65486, 16 }, { 65487, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 1017, 10 }, { 65488, 16 }, { 65489, 16 }, { 65490, 16 }, { 65491, 16 }, { 65492, 16 }, { 65493, 16 }, { 65494, 16 }, { 65495, 16 }, { 65496, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 1018, 10 }, { 65497, 16 }, { 65498, 16 }, { 65499, 16 }, { 65500, 16 }, { 65501, 16 }, { 65502, 16 }, { 65503, 16 }, { 65504, 16 }, { 65505, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 2040, 11 }, { 65506, 16 }, { 65507, 16 }, { 65508, 16 }, { 65509, 16 }, { 65510, 16 }, { 65511, 16 }, { 65512, 16 }, { 65513, 16 }, { 65514, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 65515, 16 }, { 65516, 16 }, { 65517, 16 }, { 65518, 16 }, { 65519, 16 }, { 65520, 16 }, { 65521, 16 }, { 65522, 16 }, { 65523, 16 }, { 65524, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 2041, 11 }, { 65525, 16 }, { 65526, 16 }, { 65527, 16 }, { 65528, 16 }, { 65529, 16 }, { 65530, 16 }, { 65531, 16 }, { 65532, 16 }, { 65533, 16 }, { 65534, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 } + }; + static unsigned short const UVAC_HT[256][2] = { + { 0, 2 }, { 1, 2 }, { 4, 3 }, { 10, 4 }, { 24, 5 }, { 25, 5 }, { 56, 6 }, { 120, 7 }, { 500, 9 }, { 1014, 10 }, { 4084, 12 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 11, 4 }, { 57, 6 }, { 246, 8 }, { 501, 9 }, { 2038, 11 }, { 4085, 12 }, { 65416, 16 }, { 65417, 16 }, { 65418, 16 }, { 65419, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 26, 5 }, { 247, 8 }, { 1015, 10 }, { 4086, 12 }, { 32706, 15 }, { 65420, 16 }, { 65421, 16 }, { 65422, 16 }, { 65423, 16 }, { 65424, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 27, 5 }, { 248, 8 }, { 1016, 10 }, { 4087, 12 }, { 65425, 16 }, { 65426, 16 }, { 65427, 16 }, { 65428, 16 }, { 65429, 16 }, { 65430, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 58, 6 }, { 502, 9 }, { 65431, 16 }, { 65432, 16 }, { 65433, 16 }, { 65434, 16 }, { 65435, 16 }, { 65436, 16 }, { 65437, 16 }, { 65438, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 59, 6 }, { 1017, 10 }, { 65439, 16 }, { 65440, 16 }, { 65441, 16 }, { 65442, 16 }, { 65443, 16 }, { 65444, 16 }, { 65445, 16 }, { 65446, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 121, 7 }, { 2039, 11 }, { 65447, 16 }, { 65448, 16 }, { 65449, 16 }, { 65450, 16 }, { 65451, 16 }, { 65452, 16 }, { 65453, 16 }, { 65454, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 122, 7 }, { 2040, 11 }, { 65455, 16 }, { 65456, 16 }, { 65457, 16 }, { 65458, 16 }, { 65459, 16 }, { 65460, 16 }, { 65461, 16 }, { 65462, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 249, 8 }, { 65463, 16 }, { 65464, 16 }, { 65465, 16 }, { 65466, 16 }, { 65467, 16 }, { 65468, 16 }, { 65469, 16 }, { 65470, 16 }, { 65471, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 503, 9 }, { 65472, 16 }, { 65473, 16 }, { 65474, 16 }, { 65475, 16 }, { 65476, 16 }, { 65477, 16 }, { 65478, 16 }, { 65479, 16 }, { 65480, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 504, 9 }, { 65481, 16 }, { 65482, 16 }, { 65483, 16 }, { 65484, 16 }, { 65485, 16 }, { 65486, 16 }, { 65487, 16 }, { 65488, 16 }, { 65489, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 505, 9 }, { 65490, 16 }, { 65491, 16 }, { 65492, 16 }, { 65493, 16 }, { 65494, 16 }, { 65495, 16 }, { 65496, 16 }, { 65497, 16 }, { 65498, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 506, 9 }, { 65499, 16 }, { 65500, 16 }, { 65501, 16 }, { 65502, 16 }, { 65503, 16 }, { 65504, 16 }, { 65505, 16 }, { 65506, 16 }, { 65507, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 2041, 11 }, { 65508, 16 }, { 65509, 16 }, { 65510, 16 }, { 65511, 16 }, { 65512, 16 }, { 65513, 16 }, { 65514, 16 }, { 65515, 16 }, { 65516, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 16352, 14 }, { 65517, 16 }, { 65518, 16 }, { 65519, 16 }, { 65520, 16 }, { 65521, 16 }, { 65522, 16 }, { 65523, 16 }, { 65524, 16 }, { 65525, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, + { 1018, 10 }, { 32707, 15 }, { 65526, 16 }, { 65527, 16 }, { 65528, 16 }, { 65529, 16 }, { 65530, 16 }, { 65531, 16 }, { 65532, 16 }, { 65533, 16 }, { 65534, 16 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 } + }; + static int const YQT[] = { 16, 11, 10, 16, 24, 40, 51, 61, 12, 12, 14, 19, 26, 58, 60, 55, 14, 13, 16, 24, 40, 57, 69, 56, 14, 17, 22, 29, 51, 87, 80, 62, 18, 22, + 37, 56, 68, 109, 103, 77, 24, 35, 55, 64, 81, 104, 113, 92, 49, 64, 78, 87, 103, 121, 120, 101, 72, 92, 95, 98, 112, 100, 103, 99 }; + static int const UVQT[] = { 17, 18, 24, 47, 99, 99, 99, 99, 18, 21, 26, 66, 99, 99, 99, 99, 24, 26, 56, 99, 99, 99, 99, 99, 47, 66, 99, 99, 99, 99, 99, 99, + 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99, 99 }; + static float const aasf[] = { 1.0f * 2.828427125f, 1.387039845f * 2.828427125f, 1.306562965f * 2.828427125f, 1.175875602f * 2.828427125f, + 1.0f * 2.828427125f, 0.785694958f * 2.828427125f, 0.541196100f * 2.828427125f, 0.275899379f * 2.828427125f }; + + int row, col, i, k, subsample; + float fdtbl_Y[64], fdtbl_UV[64]; + unsigned char YTable[64], UVTable[64]; + + if (!data || !width || !height || comp > 4 || comp < 1) { + return 0; + } + + quality = quality ? quality : 90; + subsample = quality <= 90 ? 1 : 0; + quality = quality < 1 ? 1 : quality > 100 ? 100 + : quality; + quality = quality < 50 ? 5000 / quality : 200 - quality * 2; + + for (i = 0; i < 64; ++i) { + int uvti, yti = (YQT[i] * quality + 50) / 100; + YTable[stbiw__jpg_ZigZag[i]] = (unsigned char)(yti < 1 ? 1 : yti > 255 ? 255 + : yti); + uvti = (UVQT[i] * quality + 50) / 100; + UVTable[stbiw__jpg_ZigZag[i]] = (unsigned char)(uvti < 1 ? 1 : uvti > 255 ? 255 + : uvti); + } + + for (row = 0, k = 0; row < 8; ++row) { + for (col = 0; col < 8; ++col, ++k) { + fdtbl_Y[k] = 1 / (YTable[stbiw__jpg_ZigZag[k]] * aasf[row] * aasf[col]); + fdtbl_UV[k] = 1 / (UVTable[stbiw__jpg_ZigZag[k]] * aasf[row] * aasf[col]); + } + } + + // Write Headers + { + static unsigned char const head0[] = { 0xFF, 0xD8, 0xFF, 0xE0, 0, 0x10, 'J', 'F', 'I', 'F', 0, 1, 1, 0, 0, 1, 0, 1, 0, 0, 0xFF, 0xDB, 0, 0x84, 0 }; + static unsigned char const head2[] = { 0xFF, 0xDA, 0, 0xC, 3, 1, 0, 2, 0x11, 3, 0x11, 0, 0x3F, 0 }; + unsigned char const head1[] = { 0xFF, 0xC0, 0, 0x11, 8, (unsigned char)(height >> 8), STBIW_UCHAR(height), (unsigned char)(width >> 8), STBIW_UCHAR(width), + 3, 1, (unsigned char)(subsample ? 0x22 : 0x11), 0, 2, 0x11, 1, 3, 0x11, 1, 0xFF, 0xC4, 0x01, 0xA2, 0 }; + s->func(s->context, (void*)head0, sizeof(head0)); + s->func(s->context, (void*)YTable, sizeof(YTable)); + stbiw__putc(s, 1); + s->func(s->context, UVTable, sizeof(UVTable)); + s->func(s->context, (void*)head1, sizeof(head1)); + s->func(s->context, (void*)(std_dc_luminance_nrcodes + 1), sizeof(std_dc_luminance_nrcodes) - 1); + s->func(s->context, (void*)std_dc_luminance_values, sizeof(std_dc_luminance_values)); + stbiw__putc(s, 0x10); // HTYACinfo + s->func(s->context, (void*)(std_ac_luminance_nrcodes + 1), sizeof(std_ac_luminance_nrcodes) - 1); + s->func(s->context, (void*)std_ac_luminance_values, sizeof(std_ac_luminance_values)); + stbiw__putc(s, 1); // HTUDCinfo + s->func(s->context, (void*)(std_dc_chrominance_nrcodes + 1), sizeof(std_dc_chrominance_nrcodes) - 1); + s->func(s->context, (void*)std_dc_chrominance_values, sizeof(std_dc_chrominance_values)); + stbiw__putc(s, 0x11); // HTUACinfo + s->func(s->context, (void*)(std_ac_chrominance_nrcodes + 1), sizeof(std_ac_chrominance_nrcodes) - 1); + s->func(s->context, (void*)std_ac_chrominance_values, sizeof(std_ac_chrominance_values)); + s->func(s->context, (void*)head2, sizeof(head2)); + } + + // Encode 8x8 macroblocks + { + static unsigned short const fillBits[] = { 0x7F, 7 }; + int DCY = 0, DCU = 0, DCV = 0; + int bitBuf = 0, bitCnt = 0; + // comp == 2 is grey+alpha (alpha is ignored) + int ofsG = comp > 2 ? 1 : 0, ofsB = comp > 2 ? 2 : 0; + unsigned char const* dataR = (unsigned char const*)data; + unsigned char const* dataG = dataR + ofsG; + unsigned char const* dataB = dataR + ofsB; + int x, y, pos; + if (subsample) { + for (y = 0; y < height; y += 16) { + for (x = 0; x < width; x += 16) { + float Y[256], U[256], V[256]; + for (row = y, pos = 0; row < y + 16; ++row) { + // row >= height => use last input row + int clamped_row = (row < height) ? row : height - 1; + int base_p = (stbi__flip_vertically_on_write ? (height - 1 - clamped_row) : clamped_row) * width * comp; + for (col = x; col < x + 16; ++col, ++pos) { + // if col >= width => use pixel from last input column + int p = base_p + ((col < width) ? col : (width - 1)) * comp; + float r = dataR[p], g = dataG[p], b = dataB[p]; + Y[pos] = +0.29900f * r + 0.58700f * g + 0.11400f * b - 128; + U[pos] = -0.16874f * r - 0.33126f * g + 0.50000f * b; + V[pos] = +0.50000f * r - 0.41869f * g - 0.08131f * b; + } + } + DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y + 0, 16, fdtbl_Y, DCY, YDC_HT, YAC_HT); + DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y + 8, 16, fdtbl_Y, DCY, YDC_HT, YAC_HT); + DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y + 128, 16, fdtbl_Y, DCY, YDC_HT, YAC_HT); + DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y + 136, 16, fdtbl_Y, DCY, YDC_HT, YAC_HT); + + // subsample U,V + { + float subU[64], subV[64]; + int yy, xx; + for (yy = 0, pos = 0; yy < 8; ++yy) { + for (xx = 0; xx < 8; ++xx, ++pos) { + int j = yy * 32 + xx * 2; + subU[pos] = (U[j + 0] + U[j + 1] + U[j + 16] + U[j + 17]) * 0.25f; + subV[pos] = (V[j + 0] + V[j + 1] + V[j + 16] + V[j + 17]) * 0.25f; + } + } + DCU = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, subU, 8, fdtbl_UV, DCU, UVDC_HT, UVAC_HT); + DCV = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, subV, 8, fdtbl_UV, DCV, UVDC_HT, UVAC_HT); + } + } + } + } else { + for (y = 0; y < height; y += 8) { + for (x = 0; x < width; x += 8) { + float Y[64], U[64], V[64]; + for (row = y, pos = 0; row < y + 8; ++row) { + // row >= height => use last input row + int clamped_row = (row < height) ? row : height - 1; + int base_p = (stbi__flip_vertically_on_write ? (height - 1 - clamped_row) : clamped_row) * width * comp; + for (col = x; col < x + 8; ++col, ++pos) { + // if col >= width => use pixel from last input column + int p = base_p + ((col < width) ? col : (width - 1)) * comp; + float r = dataR[p], g = dataG[p], b = dataB[p]; + Y[pos] = +0.29900f * r + 0.58700f * g + 0.11400f * b - 128; + U[pos] = -0.16874f * r - 0.33126f * g + 0.50000f * b; + V[pos] = +0.50000f * r - 0.41869f * g - 0.08131f * b; + } + } + + DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y, 8, fdtbl_Y, DCY, YDC_HT, YAC_HT); + DCU = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, U, 8, fdtbl_UV, DCU, UVDC_HT, UVAC_HT); + DCV = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, V, 8, fdtbl_UV, DCV, UVDC_HT, UVAC_HT); + } + } + } + + // Do the bit alignment of the EOI marker + stbiw__jpg_writeBits(s, &bitBuf, &bitCnt, fillBits); + } + + // EOI + stbiw__putc(s, 0xFF); + stbiw__putc(s, 0xD9); + + return 1; +} + +STBIWDEF int stbi_write_jpg_to_func(stbi_write_func* func, void* context, int x, int y, int comp, void const* data, int quality) +{ + stbi__write_context s = { 0 }; + stbi__start_write_callbacks(&s, func, context); + return stbi_write_jpg_core(&s, x, y, comp, (void*)data, quality); +} + +# ifndef STBI_WRITE_NO_STDIO +STBIWDEF int stbi_write_jpg(char const* filename, int x, int y, int comp, void const* data, int quality) +{ + stbi__write_context s = { 0 }; + if (stbi__start_write_file(&s, filename)) { + int r = stbi_write_jpg_core(&s, x, y, comp, data, quality); + stbi__end_write_file(&s); + return r; + } else + return 0; +} +# endif + +#endif // STB_IMAGE_WRITE_IMPLEMENTATION + +/* Revision history + 1.16 (2021-07-11) + make Deflate code emit uncompressed blocks when it would otherwise expand + support writing BMPs with alpha channel + 1.15 (2020-07-13) unknown + 1.14 (2020-02-02) updated JPEG writer to downsample chroma channels + 1.13 + 1.12 + 1.11 (2019-08-11) + + 1.10 (2019-02-07) + support utf8 filenames in Windows; fix warnings and platform ifdefs + 1.09 (2018-02-11) + fix typo in zlib quality API, improve STB_I_W_STATIC in C++ + 1.08 (2018-01-29) + add stbi__flip_vertically_on_write, external zlib, zlib quality, choose PNG filter + 1.07 (2017-07-24) + doc fix + 1.06 (2017-07-23) + writing JPEG (using Jon Olick's code) + 1.05 ??? + 1.04 (2017-03-03) + monochrome BMP expansion + 1.03 ??? + 1.02 (2016-04-02) + avoid allocating large structures on the stack + 1.01 (2016-01-16) + STBIW_REALLOC_SIZED: support allocators with no realloc support + avoid race-condition in crc initialization + minor compile issues + 1.00 (2015-09-14) + installable file IO function + 0.99 (2015-09-13) + warning fixes; TGA rle support + 0.98 (2015-04-08) + added STBIW_MALLOC, STBIW_ASSERT etc + 0.97 (2015-01-18) + fixed HDR asserts, rewrote HDR rle logic + 0.96 (2015-01-17) + add HDR output + fix monochrome BMP + 0.95 (2014-08-17) + add monochrome TGA output + 0.94 (2014-05-31) + rename private functions to avoid conflicts with stb_image.h + 0.93 (2014-05-27) + warning fixes + 0.92 (2010-08-01) + casts to unsigned char to fix warnings + 0.91 (2010-07-17) + first public release + 0.90 first internal release +*/ + +/* +------------------------------------------------------------------------------ +This software is available under 2 licenses -- choose whichever you prefer. +------------------------------------------------------------------------------ +ALTERNATIVE A - MIT License +Copyright (c) 2017 Sean Barrett +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +of the Software, and to permit persons to whom the Software is furnished to do +so, subject to the following conditions: +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. +------------------------------------------------------------------------------ +ALTERNATIVE B - Public Domain (www.unlicense.org) +This is free and unencumbered software released into the public domain. +Anyone is free to copy, modify, publish, use, compile, sell, or distribute this +software, either in source code form or as a compiled binary, for any purpose, +commercial or non-commercial, and by any means. +In jurisdictions that recognize copyright laws, the author or authors of this +software dedicate any and all copyright interest in the software to the public +domain. We make this dedication for the benefit of the public at large and to +the detriment of our heirs and successors. We intend this dedication to be an +overt act of relinquishment in perpetuity of all present and future rights to +this software under copyright law. +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN +ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION +WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +------------------------------------------------------------------------------ +*/ diff --git a/src/libtemple/ioport.h b/src/libtemple/ioport.h new file mode 100644 index 0000000..69e69e3 --- /dev/null +++ b/src/libtemple/ioport.h @@ -0,0 +1,6 @@ +u8 ioport_read_u8(u16 address); +u16 ioport_read_u16(u16 address); +u32 ioport_read_u32(u16 address); +void ioport_write_u8(u16 address, u8 value); +void ioport_write_u16(u16 address, u16 value); +void ioport_write_u32(u16 address, u32 value); diff --git a/src/libtemple/libtemple.cpp b/src/libtemple/libtemple.cpp new file mode 100644 index 0000000..82ce415 --- /dev/null +++ b/src/libtemple/libtemple.cpp @@ -0,0 +1,77 @@ +unsigned long ioport_read_u8(unsigned short address) { return 0; } + +unsigned long ioport_read_u16(unsigned short address) { return 0; } + +unsigned long ioport_read_u32(unsigned short address) { return 0; } + +void ioport_write_u8(unsigned short address, unsigned char value) { } + +void ioport_write_u16(unsigned short address, unsigned short value) { } + +void ioport_write_u32(unsigned short address, unsigned int value) { } + +bool os_blink(char const* frequency_as_string) { return 0; } + +unsigned long os_call(unsigned long function_name, unsigned long arg) { return 0; } + +unsigned int os_device_calloc(unsigned int size) { return 0; } + +void os_exit() { } + +char const* os_file_picker(char const* path, char const* glob) { return 0; } + +char const* os_files_list(char const* path) { return 0; } + +bool os_is_vm() { return 0; } + +bool os_path_exists(char const* path) { return 0; } + +void os_pc_speaker(char const* frequency_as_string) { } + +unsigned long os_random() { return 0; } + +unsigned long os_read_entire_file(char const* filename, long* size) +{ + return 0; +} + +void os_screenshot() { } + +char const* os_to_uppercase(char const* input_string) { return 0; } + +void os_write_entire_file(char const* filename, unsigned char* buffer, + long size) { } + +long pci_find(long class_code) { return 0; } + +unsigned long pci_read_u8(long bus, long device, long fun, long offset) +{ + return 0; +} + +unsigned long pci_read_u16(long bus, long device, long fun, long offset) +{ + return 0; +} + +unsigned long pci_read_u32(long bus, long device, long fun, long offset) +{ + return 0; +} + +void pci_write_u8(long bus, long device, long fun, long offset, + unsigned char value) { } + +void pci_write_u16(long bus, long device, long fun, long offset, + unsigned short value) { } + +void pci_write_u32(long bus, long device, long fun, long offset, + unsigned int value) { } + +void time_busy(long duration) { } + +long time_jiffies() { return 0; } + +long time_now() { return 0; } + +void time_sleep(long duration) { } \ No newline at end of file diff --git a/src/libtemple/os.h b/src/libtemple/os.h new file mode 100644 index 0000000..24e489e --- /dev/null +++ b/src/libtemple/os.h @@ -0,0 +1,15 @@ +bool os_blink(char const* frequency_as_string); +unsigned long os_call(unsigned long function_name, unsigned long arg); +unsigned int os_device_calloc(unsigned int size); +void os_exit(); +char const* os_file_picker(char const* path, char const* glob); +char const* os_files_list(char const* path); +bool os_is_vm(); +bool os_path_exists(char const* path); +void os_pc_speaker(char const* frequency_as_string); +unsigned long os_random(); +u8* os_read_entire_file(char const* filename, i64* size); +void os_screenshot(); +char const* os_to_uppercase(char const* input_string); +void os_write_entire_file(char const* filename, unsigned char* buffer, + i64 size); \ No newline at end of file diff --git a/src/libtemple/pci.h b/src/libtemple/pci.h new file mode 100644 index 0000000..33d7fea --- /dev/null +++ b/src/libtemple/pci.h @@ -0,0 +1,10 @@ +long pci_find(long class_code); +unsigned long pci_read_u8(long bus, long device, long fun, long offset); +unsigned long pci_read_u16(long bus, long device, long fun, long offset); +unsigned long pci_read_u32(long bus, long device, long fun, long offset); +void pci_write_u8(long bus, long device, long fun, long offset, + unsigned char value); +void pci_write_u16(long bus, long device, long fun, long offset, + unsigned short value); +void pci_write_u32(long bus, long device, long fun, long offset, + unsigned int value); \ No newline at end of file diff --git a/src/libtemple/time.h b/src/libtemple/time.h new file mode 100644 index 0000000..e2fe7c4 --- /dev/null +++ b/src/libtemple/time.h @@ -0,0 +1,4 @@ +void time_busy(i64 duration); +i64 time_jiffies(); +i64 time_now(); +void time_sleep(i64 duration); \ No newline at end of file diff --git a/src/net/devices/virtio.h b/src/net/devices/virtio.h new file mode 100644 index 0000000..989e1d0 --- /dev/null +++ b/src/net/devices/virtio.h @@ -0,0 +1,28 @@ +struct virtio_queue_buf { + u64 address; + u32 length; + u16 flags; + u16 next; +}; +struct virtio_avail { + u16 flags; + u16 index; + u16 ring[256]; + u16 int_index; +}; +struct virtio_used_item { + u32 index; + u32 length; +}; +struct virtio_used { + u16 flags; + u16 index; + virtio_used_item ring[256]; + u16 int_index; +}; +struct virtio_queue { + virtio_queue_buf buffers[256]; + virtio_avail available; + u8 padding[3578]; + virtio_used used; +}; \ No newline at end of file diff --git a/src/net/devices/virtio.jakt b/src/net/devices/virtio.jakt new file mode 100644 index 0000000..647af28 --- /dev/null +++ b/src/net/devices/virtio.jakt @@ -0,0 +1,143 @@ +import relative parent::os::os { OS } +import relative parent::os::pci { PCI, PCIDevice } + +enum VirtIOConfig: u8 { + acknowledge = 1 + driver = 2 + driver_ok = 4 +} + +enum VirtIOReg: u16 { + host_features = 0 + guest_features = 4 + queue_page_frame_number = 8 + queue_size = 12 + queue_select = 14 + queue_notify = 16 + status = 18 + isr = 19 + config = 20 +} + +class VirtIO { + public pci_device: PCIDevice + public rq_index: i64 + public rq_size: u16 + public rq: u32 + public tq_size: u16 + public tq: u32 + public fn rx_frame(mut this) throws -> [u8] { + mut frame: [u8] = [] + mut queue_notify: bool = false + unsafe { + cpp { +" +#include <../../src/net/devices/virtio.h> +virtio_queue *rq = (virtio_queue*)this->rq; +i64 i = this->rq_index; +i64 used_index = rq->used.index; +if (used_index < i) + used_index += 0x10000; +if (used_index && i != used_index) { + virtio_used_item* item = rq->used.ring; + u8* buffer = (u8*)rq->buffers[item[i % 256].index + 1].address; + i64 length = item[i % 256].length - 10; + for (i64 j = 0; j < length; j++) + frame.push(buffer[j]); + this->rq_index = used_index % 0x10000; + rq->available.index++; + queue_notify = true; +} +" + } + } + if queue_notify { + .pci_device.io_write_u16(offset: VirtIOReg::queue_notify as! u16, value: 0) + } + return frame + } + public fn tx_frame(mut this, anon mut data: [u8]) throws { + mut size = data.size() + unsafe { + cpp { +" +#include <../../src/net/devices/virtio.h> +virtio_queue *tq = (virtio_queue*)this->tq; +int tq_idx = tq->available.index % 256; +int tq_idx2 = tq_idx % 128; +memset((u8*)tq->buffers[tq_idx2 * 2].address, 0, 10); +u8 *buffer = (u8*)tq->buffers[(tq_idx2 * 2) + 1].address; + for (int i = 0; i < size; i++) + buffer[i] = data[i]; +tq->buffers[tq_idx2 * 2].length = 10; +tq->buffers[tq_idx2 * 2].flags = 1; +tq->buffers[tq_idx2 * 2].next = (tq_idx2 * 2) + 1; +tq->buffers[(tq_idx2 * 2) + 1].length = size; +tq->buffers[(tq_idx2 * 2) + 1].flags = 0; +tq->buffers[(tq_idx2 * 2) + 1].next = 0; +tq->available.ring[tq_idx] = tq_idx2 * 2; +tq->available.index++; +" + } + } + .pci_device.io_write_u16(offset: VirtIOReg::queue_notify as! u16, value: 1) + } + fn reset_device(this) { + .pci_device.io_write_u8(offset: VirtIOReg::status as! u16, value: 0) + } + fn found_driver(this) throws { + .pci_device.io_write_u8(offset: VirtIOReg::status as! u16, + value: .pci_device.io_read_u8(VirtIOReg::status as! u16) | VirtIOConfig::acknowledge as! u8 | VirtIOConfig::driver as! u8) + } + fn setup_rx_queue(mut this) throws { + .pci_device.io_write_u16(offset: VirtIOReg::queue_select as! u16, value: 0) + .rq_size = .pci_device.io_read_u16(VirtIOReg::queue_size as! u16) + .rq = OS::device_calloc(16384) + .pci_device.io_write_u32(offset: VirtIOReg::queue_page_frame_number as! u16, value: .rq / 4096) + } + fn setup_tx_queue(mut this) throws { + .pci_device.io_write_u16(offset: VirtIOReg::queue_select as! u16, value: 1) + .tq_size = .pci_device.io_read_u16(VirtIOReg::queue_size as! u16) + .tq = OS::device_calloc(16384) + .pci_device.io_write_u32(offset: VirtIOReg::queue_page_frame_number as! u16, value: .tq / 4096) + } + fn init_queue_buffers(this) { + unsafe { + cpp { +" +#include <../../src/net/devices/virtio.h> +virtio_queue *rq = (virtio_queue*)this->rq; +virtio_queue *tq = (virtio_queue*)this->tq; +for (int i = 0; i < 128; i++) { + rq->buffers[i * 2].address = (u64)calloc(1, 16); + rq->buffers[i * 2].length = 10; + rq->buffers[i * 2].flags = 3; + rq->buffers[i * 2].next = (i * 2) + 1; + rq->buffers[(i * 2) + 1].address = (u64)calloc(1, 2048); + rq->buffers[(i * 2) + 1].length = 2048; + rq->buffers[(i * 2) + 1].flags = 2; + rq->buffers[(i * 2) + 1].next = 0; + rq->available.ring[i] = i * 2; + rq->available.ring[i + 128] = i * 2; + tq->buffers[i * 2].address = (u64)calloc(1, 16); + tq->buffers[(i * 2) + 1].address = (u64)calloc(1, 2048); +} +rq->available.index = 1; +" + } + } + } + fn init_ok(this) throws { + .pci_device.io_write_u8(offset: VirtIOReg::status as! u16, + value: .pci_device.io_read_u8(VirtIOReg::status as! u16) | VirtIOConfig::driver_ok as! u8) + .pci_device.io_write_u16(offset: VirtIOReg::queue_notify as! u16, value: 0) + } + public fn init(mut this) throws { + .reset_device() + .found_driver() + .setup_rx_queue() + .setup_tx_queue() + .init_queue_buffers() + .init_ok() + } +} \ No newline at end of file diff --git a/src/net/lib/json.jakt b/src/net/lib/json.jakt new file mode 100644 index 0000000..5cdd177 --- /dev/null +++ b/src/net/lib/json.jakt @@ -0,0 +1,350 @@ +/// Expect: +/// - output: "JsonValue::JsonArray([JsonValue::Object([\"id\": JsonValue::Number(0.5), \"displayName\": JsonValue::JsonString(\"Air\"), \"name\": JsonValue::JsonString(\"air\"), \"hardness\": JsonValue::Number(3.9), \"resistance\": JsonValue::Number(0), \"minStateId\": JsonValue::Number(0), \"maxStateId\": JsonValue::Number(0), \"states\": JsonValue::JsonArray([])])])\n" + +enum JsonValue { + Null + Bool(bool) + Number(f64) + // FIXME: This variant should be called String + JsonString(String) + // FIXME: This variant should be called Array + JsonArray([JsonValue]) + Object([String:JsonValue]) +} + +fn is_whitespace(anon c: u8) -> bool { + return match c { + b'\t' | b'\n' | b'\r' | b' ' => true + else => false + } +} + +class JsonParser { + input: String + index: usize + + public fn construct(input: String) throws -> JsonParser { + return JsonParser(input, index: 0) + } + + fn eof(this) -> bool { + return .index >= .input.length() + } + + public fn parse(mut this) throws -> JsonValue { + // FIXME: Jakt::JsonParser ignores trailing whitespace for some reason. + let value = .parse_helper() + if not .eof() { + // FIXME: "Didn't consume all input" + throw Error::from_errno(9000) + } + return value + } + + fn skip_whitespace(mut this) { + while not .eof() { + if not is_whitespace(.input.byte_at(.index)) { + break + } + .index++ + } + } + + fn consume_and_unescape_string(mut this) throws -> String { + if not .consume_specific(b'"') { + // FIXME: "Expected '"' + throw Error::from_errno(9007) + } + + mut builder = StringBuilder::create() + + loop { + mut ch = 0u8 + mut peek_index = .index + while peek_index < .input.length() { + ch = .input.byte_at(peek_index) + if ch == b'"' or ch == b'\\' { + break + } + // FIXME: This is is_ascii_c0_control() + if ch < 0x20 { + // FIXME: "Error while parsing string" + throw Error::from_errno(9008) + } + peek_index++ + } + + while peek_index != .index { + builder.append(.input.byte_at(.index)) + .index++ + } + + if .eof() { + break + } + + if ch == b'"' { + break + } + + if ch != b'\\' { + builder.append(.consume()) + continue + } + + .ignore() + + match .peek() { + b'"' | b'/' | b'\\' | b'n' | b'r' | b't' | b'b' | b'f' => { + let ch = .consume() + builder.append(match ch { + b'n' => b'\n' + b'r' => b'\r' + b't' => b'\t' + b'b' => b'\b' + b'f' => b'\f' + else => ch + }) + } + b'u' => { + eprintln("FIXME: Implement unicode literals") + abort() + } + else => { + // FIXME: "Error while parsing string" + throw Error::from_errno(9009) + } + } + } + + if not .consume_specific(b'"') { + // FIXME: "Expected '"'" + throw Error::from_errno(9010) + } + + return builder.to_string() + } + + fn ignore(mut this) { + .index++ + } + + fn peek(this) -> u8 { + if .eof() { + return 0 + } + return .input.byte_at(.index) + } + + fn consume(mut this) -> u8 { + let ch = .peek() + .index++ + return ch + } + + fn consume_specific(mut this, anon expected: u8) -> bool { + if .peek() != expected { + return false + } + .index++ + return true + } + + fn parse_helper(mut this) throws -> JsonValue { + .skip_whitespace() + return match .peek() { + b'{' => .parse_object() + b'[' => .parse_array() + b'"' => .parse_string() + b'-' => .parse_number() + b'0' | b'1' | b'2' | b'3' | b'4' | b'5' | b'6' | b'7' | b'8' | b'9' => .parse_number() + b'f' => .parse_false() + b't' => .parse_true() + b'n' => .parse_null() + else => .parse_failure(error_message: "Unexpected character") + } + } + + fn parse_failure(this, error_message: String) throws -> JsonValue { + throw Error::from_errno(9001) + } + + fn parse_array(mut this) throws -> JsonValue { + mut array: [JsonValue] = [] + if (not .consume_specific(b'[')) { + // Expected '[' + throw Error::from_errno(9014) + } + loop { + .skip_whitespace() + if .peek() == b']' { + break + } + array.push(.parse_helper()) + .skip_whitespace() + if .peek() == b']' { + break + } + if not .consume_specific(b',') { + // Expected ',' + throw Error::from_errno(9014) + } + .skip_whitespace() + if .peek() == b']' { + // Unexpected ']' + throw Error::from_errno(9014) + } + } + if not .consume_specific(b']') { + // Expected ']' + throw Error::from_errno(9015) + } + return JsonValue::JsonArray(array) + } + + fn parse_object(mut this) throws -> JsonValue { + if not .consume_specific(b'{') { + // FIXME: "Expected '{'" + throw Error::from_errno(9002) + } + + mut values: [String:JsonValue] = [:] + + loop { + .skip_whitespace() + if .peek() == b'}' { + break + } + .skip_whitespace() + let key = .consume_and_unescape_string() + .skip_whitespace() + if not .consume_specific(b':') { + // FIXME: "Expected ':'" + throw Error::from_errno(9003) + } + .skip_whitespace() + let value = .parse_helper() + // FIXME: This should say `values[key] = value`, but the compiler doesn't wrap it in TRY() + values.set(key, value) + .skip_whitespace() + if .peek() == b'}' { + break + } + if not .consume_specific(b',') { + // FIXME: "Expected ','" + throw Error::from_errno(9004) + } + .skip_whitespace() + if .peek() == b'}' { + // FIXME: "Unexpected '}'" + throw Error::from_errno(9005) + } + } + if not .consume_specific(b'}') { + // FIXME: "Expected '}'" + throw Error::from_errno(9006) + } + return JsonValue::Object(values) + } + + fn char_to_f64(anon num: u8) throws -> f64 { + // FIXME 1: Shouldn't need this function at all + // FIXME 2: Shouldn't need return in else branch + return match num { + 0u8 => 0.0 + 1u8 => 1.0 + 2u8 => 2.0 + 3u8 => 3.0 + 4u8 => 4.0 + 5u8 => 5.0 + 6u8 => 6.0 + 7u8 => 7.0 + 8u8 => 8.0 + 9u8 => 9.0 + else => { + // FIXME: "Unexpected number" + throw Error::from_errno(9017) + } + } + } + + fn parse_number(mut this) throws -> JsonValue { + // FIXME: This implementation doesn't match JsonParser.cpp + let is_negative = .consume_specific(b'-') + mut decimal_start_index: usize? = None + + mut value = 0.0 + + while not .eof() { + let ch = .peek() + if ch == b'.' { + if decimal_start_index.has_value() { + // FIXME: "Unexpected '.'" + throw Error::from_errno(9016) + } + decimal_start_index = .index++ + continue + } else if not (ch >= b'0' and ch <= b'9') { + break + } + + if not decimal_start_index.has_value() { + value *= 10.0 + value += char_to_f64(ch - b'0') + } else { + mut num = char_to_f64(ch - b'0') + // FIXME: This should really be: `value += pow(10, -decimal_place)*num`, but: there's no pow function and you can't multiply float by usize + let decimal_place = .index - decimal_start_index.value() + for i in 0..decimal_place { + num /= 10.0 + } + value += num + } + .index++ + } + + if is_negative { + value *= -1.0 + } + + return JsonValue::Number(value) + } + + fn parse_string(mut this) throws -> JsonValue { + return JsonValue::JsonString(.consume_and_unescape_string()) + } + + fn parse_false(mut this) throws -> JsonValue { + if (.consume() != b'f' or .consume() != b'a' or .consume() != b'l' or .consume() != b's' or .consume() != b'e') { + // FIXME: "Expected 'false'" + throw Error::from_errno(9011) + } + return JsonValue::Bool(false) + } + + fn parse_true(mut this) throws -> JsonValue { + if (.consume() != b't' or .consume() != b'r' or .consume() != b'u' or .consume() != b'e') { + // FIXME: "Expected 'true'" + throw Error::from_errno(9012) + } + return JsonValue::Bool(true) + } + + fn parse_null(mut this) throws -> JsonValue { + if (.consume() != b'n' or .consume() != b'u' or .consume() != b'l' or .consume() != b'l') { + // FIXME: "Expected 'null'" + throw Error::from_errno(9013) + } + return JsonValue::Null + } +} + +// fn parse_json(input: String) throws -> JsonValue { +// mut parser = JsonParser::construct(input) +// return parser.parse() +// } +// +// fn main() { +// let value = parse_json(input: "[{\"id\":0.5,\"displayName\":\"Air\",\"name\":\"air\",\"hardness\":3.9,\"resistance\":0,\"minStateId\":0,\"maxStateId\":0,\"states\":[]}]") +// println("{}", value) +// } \ No newline at end of file diff --git a/src/net/lib/util.jakt b/src/net/lib/util.jakt new file mode 100644 index 0000000..434158d --- /dev/null +++ b/src/net/lib/util.jakt @@ -0,0 +1,147 @@ +import relative parent::os::os { OS } + +struct Util { + fn get_address_u32_from_ipv4_u8_array(anon array: [u8]) -> u32 { + if array.size() != 4 { + return 0 + } + mut address: u32 = (array[3] as! u32 & 0xff) as! u32 + address += ((array[2] as! u32 & 0xff) << 8) as! u32 + address += ((array[1] as! u32 & 0xff) << 16) as! u32 + address += ((array[0] as! u32 & 0xff) << 24) as! u32 + return address + } + fn get_hexadecimal_string_from_ipv4_u8_array(anon array: [u8]) throws -> String { + mut s = StringBuilder::create() + unsafe { + cpp { + "char *chars = (char*)calloc(32, 1); + sprintf(chars, \"%02x%02x%02x%02x\", array[0], array[1], array[2], array[3]); + s.append_c_string(chars); + delete(chars);" + } + } + return s.to_string() + } + fn get_md5_string_from_string(anon s: String) throws -> String { + mut sb = StringBuilder::create() + unsafe { + cpp { + " + char* md5 = (char*)os_call((u64)\"@saubari_get_md5_string_from_string\", (u64)s.characters()); + sb.append_c_string(md5); + delete(md5); + " + } + } + return sb.to_string() + } + fn get_ipv4_u8_array_from_address_string(anon s: String) throws -> [u8] { + mut address: [u8] = [] + let octet_strings = s.split(c'.') + for octet_string in octet_strings { + unsafe { + cpp { + "auto value = octet_string.to_number(); + if (value.has_value()) { + auto result = value.release_value(); + address.push(result & 0xff); + }" + } + } + } + return address + } + fn get_ipv4_u8_array_from_address_u32(anon addr: u32) throws -> [u8] { + mut address: [u8] = [] + // let source_address: [u8] = [ipv4_packet[12], ipv4_packet[13], ipv4_packet[14], ipv4_packet[15]] + address.push(((addr >> 24) & 0xff) as! u8) + address.push(((addr >> 16) & 0xff) as! u8) + address.push(((addr >> 8) & 0xff) as! u8) + address.push((addr & 0xff) as! u8) + return address + } + fn get_string_from_u8_array(anon array: [u8]) throws -> String { + mut s = StringBuilder::create() + unsafe { + cpp { + "for (int i = 0; i < array.size(); i++) { + s.append(array[i]); + }" + } + } + return s.to_string() + } + fn get_u16_from_u8_array(anon array: [u8], anon offset: i64) -> u16{ + return (array[offset] as! u16 << 8) + array[offset + 1] as! u16 + } + fn get_u16_from_u8_arrayslice(anon array: ArraySlice, anon offset: i64) -> u16{ + return (array[offset] as! u16 << 8) + array[offset + 1] as! u16 + } + fn push_string_to_u8_array(anon mut array: [u8], anon s: String) throws { + for i in 0..s.length() { + unsafe { + cpp { + "array.push(s.characters()[i]);" + } + } + } + } + fn push_u16_to_u8_array(anon mut array: [u8], anon value: u16) throws { + array.push((value >> 8) as! u8) + array.push((value & 0xff) as! u8) + } + fn push_u32_to_u8_array(anon mut array: [u8], anon value: u32) throws { + mut val_u32_to_u8: u32 = 0 + val_u32_to_u8 = (value >> 24) & 0xff + array.push(val_u32_to_u8 as! u8) + val_u32_to_u8 = (value >> 16) & 0xff + array.push(val_u32_to_u8 as! u8) + val_u32_to_u8 = (value >> 8) & 0xff + array.push(val_u32_to_u8 as! u8) + array.push((value & 0xff) as! u8) + } + fn get_dictionary_from_json_file(anon json_file: String) throws -> [String:String] { + mut dictionary: [String:String] = Dictionary() + let json_bytes = OS::read_entire_file(json_file) + let json_string = get_string_from_u8_array(json_bytes) + unsafe { + cpp { + "auto json = JsonValue::from_string(json_string).value(); + auto const& object = json.as_object(); + object.for_each_member([&]([[maybe_unused]] auto& property_name, [[maybe_unused]] const JsonValue& property_value) { + dictionary.set(property_name, property_value.deprecated_to_byte_string()); + });" + } + } + return dictionary + } + fn get_dictionary_from_string(anon s: String) throws -> [String:String] { + mut dictionary: [String:String] = Dictionary() + unsafe { + cpp { + "auto json = JsonValue::from_string(s).value(); + auto const& object = json.as_object(); + object.for_each_member([&]([[maybe_unused]] auto& property_name, [[maybe_unused]] const JsonValue& property_value) { + dictionary.set(property_name, property_value.deprecated_to_byte_string()); + });" + } + } + return dictionary + } + fn string_from_file(anon filepath: String) throws -> String { + if filepath.is_empty() or not OS::path_exists(filepath) { + return "" + } + let array = OS::read_entire_file(filepath) + mut s = StringBuilder::create() + unsafe { + cpp { + "for (int i = 0; i < array.size(); i++) { + s.append(array[i]); + }" + } + } + return s.to_string() + } +} \ No newline at end of file diff --git a/src/net/net.jakt b/src/net/net.jakt new file mode 100644 index 0000000..b7bdec9 --- /dev/null +++ b/src/net/net.jakt @@ -0,0 +1,173 @@ +import devices::virtio { VirtIO, VirtIOReg } + +import lib::util { Util } + +import os::os { OS } +import os::pci { PCI, PCIDevice } +import os::time { Time } + +import tcpip { TCPIP } + +class NetDevices { + public virtio: VirtIO + public fn create(pci_device: PCIDevice) throws -> NetDevices { + return NetDevices( + virtio: VirtIO(pci_device, rq_index: 0, rq_size: 0, rq: 0, tq_size: 0, tq: 0) + ) + } +} + +class Net { + public device: NetDevices + public mac_address: [u8] + public tcpip: TCPIP + public pci_device: PCIDevice + public fn init(config: [String:String]) throws -> Net { + let pci_device = PCI::find_device_by_class_code(0x020000) + mut net = Net( + device: NetDevices::create(pci_device) + mac_address: [] + tcpip: TCPIP( + ipv4_address: Util::get_ipv4_u8_array_from_address_string(config["tcpip.ipv4_address"]) + ipv4_netmask: Util::get_ipv4_u8_array_from_address_string(config["tcpip.ipv4_netmask"]) + ipv4_network: Util::get_ipv4_u8_array_from_address_string(config["tcpip.ipv4_network"]) + ipv4_gateway: Util::get_ipv4_u8_array_from_address_string(config["tcpip.ipv4_gateway"]) + dns_server_address: Util::get_ipv4_u8_array_from_address_string(config["tcpip.ipv4_dns_server_address"]) + dns_server_port: config["tcpip.ipv4_dns_server_port"].to_number().value() as! u16 + mss_size: config["tcpip.mss_size"].to_number().value() as! u16 + tx_queue: [] + ttl: 64 + arp_cache: Dictionary() + bound_sockets: Dictionary() + dns_cache: Dictionary() + tcp_sessions: [] + pending_dns_lookups: Dictionary() + pending_dns_cached_entries: Dictionary() + pending_icmp_requests: Dictionary() + timestamp_last_arp_request: 0 + rx_bytes: 0 + rx_frames: 0 + tx_bytes: 0 + tx_frames: 0 + ) + pci_device + ) + if net.pci_device.vendor_id() == 0x1af4 and net.pci_device.device_id() == 0x1000 { + println("[net] Found device: virtio-net, QEMU") + for i in 0u16..6u16 { + net.mac_address.push(net.pci_device.io_read_u8(VirtIOReg::config as! u16 + i)) + } + net.device.virtio.init() + return net + } + println("[net] No supported vendor ids found") + OS::exit() + return net + } + fn process_ethernet_frame(mut this, anon frame: [u8]) throws { + let ethertype: u16 = (frame[12] as! u16 * 256) + frame[13] as! u16 + match ethertype { + 0x0806 => { + //println("ARP") + .tcpip.process_arp_packet(.mac_address, frame) + } + 0x0800 => { + //println("IPv4") + .tcpip.process_ipv4_packet(.mac_address, frame) + } + 0x86dd => { + //.tcpip.process_ipv6_packet(frame) + } + 0x8035 => { + //.tcpip.process_rarp_packet(frame) + } + else => { + // unsupported + } + } + } + public fn process_events(mut this) throws { + mut received_frame = .rx_frame() + if received_frame.size() > 0 { + .tcpip.rx_bytes += received_frame.size() as! u64 + .tcpip.rx_frames++ + .process_ethernet_frame(received_frame) + } + .tcpip.tcp_transmit_pending_data_for_existing_sessions() + for frame in .tcpip.tx_queue { + .tx_frame(frame) + } + if .tcpip.tx_queue.size() > 0 { + .tcpip.tx_queue.shrink(0) + } + .tcpip.tcp_process_bind_request() + .tcpip.tcp_process_client_socket_request(.mac_address) + .tcpip.tcp_process_client_received_data() + .tcpip.tcp_process_client_send_requests(.mac_address) + .tcpip.dns_process_client_request(.mac_address) + .tcpip.icmp_process_client_request(.mac_address) + .tcpip.netinfo_process_client_request(.mac_address) + } + fn rx_frame(mut this) throws -> [u8] { + mut frame: [u8] = [] + if .pci_device.vendor_id() == 0x1af4 and .pci_device.device_id() == 0x1000 { + frame = .device.virtio.rx_frame() + } + return frame + } + fn tx_frame(mut this, anon mut data: [u8]) throws { + if data.size() < 1 { + return + } + while data.size() < 60 { + data.push(0u8) + } + .tcpip.tx_bytes += data.size() as! u64 + .tcpip.tx_frames++ + if .pci_device.vendor_id() == 0x1af4 and .pci_device.device_id() == 0x1000 { + .device.virtio.tx_frame(data) + } + } +} + +fn main() { + println("$WW,1$") + mut config = Util::get_dictionary_from_json_file("M:/System/Config/Net.json") + mut net = Net::init(config) + + println("[net] PCI device is {}", net.pci_device) + print("[net] MAC address is ") + for i in 0u16..5u16 { + print("{:0>2x}:", net.mac_address[i]) + } + println("{:0>2x}", net.mac_address[5]) + print("[net] IPv4 address is ") + for i in 0u16..3u16 { + print("{:d}.", net.tcpip.ipv4_address[i]) + } + println("{:d}", net.tcpip.ipv4_address[3]) + println(" ") + + // Update the ARP cache entry for IPv4 gateway address + net.tcpip.send_arp_request(net.mac_address, net.tcpip.ipv4_gateway) + + mut prev_rx_frames = net.tcpip.rx_frames + mut prev_tx_frames = net.tcpip.tx_frames + mut prev_jiffies = Time::jiffies() + + while true { + net.process_events() + if (prev_rx_frames != net.tcpip.rx_frames) or (prev_tx_frames != net.tcpip.tx_frames) { + prev_rx_frames = net.tcpip.rx_frames + prev_tx_frames = net.tcpip.tx_frames + prev_jiffies = Time::jiffies() + } + if Time::jiffies() < prev_jiffies + 250 { + Time::sleep(0) + } else { + Time::sleep(1) + } + } + + OS::exit() +} \ No newline at end of file diff --git a/src/net/os/ioport.jakt b/src/net/os/ioport.jakt new file mode 100644 index 0000000..a01eaa2 --- /dev/null +++ b/src/net/os/ioport.jakt @@ -0,0 +1,29 @@ +import extern c "ioport.h" { + extern fn ioport_read_u8(address: u16) -> u8 + extern fn ioport_read_u16(address: u16) -> u16 + extern fn ioport_read_u32(address: u16) -> u32 + extern fn ioport_write_u8(address: u16, value: u8) + extern fn ioport_write_u16(address: u16, value: u16) + extern fn ioport_write_u32(address: u16, value: u32) +} + +struct IOPort { + fn read_u8(anon address: u16) throws -> u8 { + return ioport_read_u8(address) + } + fn read_u16(anon address: u16) throws -> u16 { + return ioport_read_u16(address) + } + fn read_u32(anon address: u16) throws -> u32 { + return ioport_read_u32(address) + } + fn write_u8(address: u16, value: u8) { + return ioport_write_u8(address, value) + } + fn write_u16(address: u16, value: u16) { + return ioport_write_u16(address, value) + } + fn write_u32(address: u16, value: u32) { + return ioport_write_u32(address, value) + } +} diff --git a/src/net/os/os.jakt b/src/net/os/os.jakt new file mode 100644 index 0000000..358ac9f --- /dev/null +++ b/src/net/os/os.jakt @@ -0,0 +1,154 @@ +import extern c "os.h" { + extern fn os_blink(frequency: raw c_char) -> bool + extern fn os_call(function_name: u64, arg: u64) -> u64 + extern fn os_device_calloc(size: u32) -> u32 + extern fn os_exit() + extern fn os_file_picker(path: raw c_char, glob: raw c_char) + extern fn os_files_list(path: raw c_char) + extern fn os_is_vm() -> bool + extern fn os_path_exists(anon path: raw c_char) -> bool + extern fn os_pc_speaker(frequency: raw c_char) + extern fn os_random() -> u64 + extern fn os_screenshot() + extern fn os_to_uppercase(anon input_string: raw c_char) -> raw c_char +} + +struct OS { + fn blink(frequency: f64 = 2.5) throws -> bool { + let frequency_as_string = format("{}", frequency) + return os_blink(frequency: frequency_as_string.c_string()) + } + fn call(anon function_name: String, anon arg: String) throws -> u64 { + mut res: u64 = 0 + unsafe { + cpp { + " + res = os_call((u64)function_name.characters(), (u64)arg.characters()); + " + } + } + return res + } + fn device_calloc(anon size: u32) throws -> u32 { + return os_device_calloc(size) + } + fn device_copy_buffer(anon buffer: [u8]) -> u32 { + mut address: u32 = 0 + mut size = buffer.size() + unsafe { + cpp { + "u8 *data = (u8*)os_device_calloc(size); + for (int i = 0; i < size; i++) + data[i] = buffer[i]; + address = (uintptr_t)data;" + } + } + return address + } + fn exit() { + os_exit() + } + fn file_picker(path: String, glob: String) throws -> String { + mut s = StringBuilder::create() + unsafe { + cpp { + "char const *chars = os_file_picker(path.characters(), glob.characters()); + s.append_c_string(chars); + delete(chars);" + } + } + return s.to_string() + } + fn files_list(path: String) throws -> [String] { + mut s = StringBuilder::create() + unsafe { + cpp { + "char const *chars = os_files_list(path.characters()); + if (chars) { + s.append_c_string(chars); + delete(chars); + }" + } + } + return s.to_string().split(c'|') + } + fn path_exists(anon path: String) -> bool { + return os_path_exists(path.c_string()) + } + fn is_vm() -> bool { + return os_is_vm() + } + fn pc_speaker(frequency: f64) throws { + let frequency_as_string = format("{}", frequency) + os_pc_speaker(frequency: frequency_as_string.c_string()) + } + fn put_char(ch: u8) { + unsafe { + cpp { + "putchar(ch);" + } + } + } + fn random() -> u64 { + return os_random() + } + fn read_entire_file(anon filename: String) throws -> [u8] { + mut size = 0 + mut buffer: [u8] = [] + unsafe { + cpp { + "u8 *data = os_read_entire_file(filename.characters(), &size); + for (int i = 0; i < size; i++) + buffer.push(data[i]); + free(data);" + } + } + return buffer + } + fn read_device_memory(address: u32, size: i64) throws -> [u8] { + mut buffer: [u8] = []; + unsafe { + cpp { + "u8 *device_memory = (u8*)address; + for (int i = 0; i < size; i++) + buffer.push(device_memory[i]);" + } + } + return buffer + } + fn read_u16_from_device_memory(anon address: u32) throws -> u16 { + mut value: u16 = 0 + unsafe { + cpp { + "value = *(u16*)address;" + } + } + return value + } + fn screenshot() { + os_screenshot() + } + fn to_uppercase(anon input_string: String) throws -> String { + mut s = StringBuilder::create() + unsafe { + cpp { + "char const *chars = os_to_uppercase(input_string.characters()); + s.append_c_string(chars); + delete(chars);" + } + } + return s.to_string() + } + fn write_entire_file(filename: String, buffer: [u8]) { + mut size = buffer.size() + unsafe { + cpp { + "unsigned char *data = (unsigned char *)malloc(size); + for (int i = 0; i < size; i++) + data[i] = buffer[i]; + os_write_entire_file(filename.characters(), data, size); + free(data);" + } + } + } +} diff --git a/src/net/os/pci.jakt b/src/net/os/pci.jakt new file mode 100644 index 0000000..6d15132 --- /dev/null +++ b/src/net/os/pci.jakt @@ -0,0 +1,103 @@ +import ioport { IOPort } + +import extern c "pci.h" { + extern fn pci_find(anon class_code: i64) -> i64 + extern fn pci_read_u8(anon bus: i64, anon device: i64, anon fun: i64, anon offset: i64) -> u8 + extern fn pci_read_u16(anon bus: i64, anon device: i64, anon fun: i64, anon offset: i64) -> u16 + extern fn pci_read_u32(anon bus: i64, anon device: i64, anon fun: i64, anon offset: i64) -> u32 + extern fn pci_write_u8(anon bus: i64, anon device: i64, anon fun: i64, anon offset: i64, anon value: u8) + extern fn pci_write_u16(anon bus: i64, anon device: i64, anon fun: i64, anon offset: i64, anon value: u16) + extern fn pci_write_u32(anon bus: i64, anon device: i64, anon fun: i64, anon offset: i64, anon value: u32) +} + +struct PCIDevice { + bus: i64 + device: i64 + fun: i64 + pci_vendor_id: u16 + pci_device_id: u16 + bar: [u32] + public fn enable_bus_master(mut this) { + .set_command(.command() | 0x4) + } + public fn read_u8(this, anon offset: i64) -> u8 { + return pci_read_u8(.bus, .device, .fun, offset) + } + public fn read_u16(this, anon offset: i64) -> u16 { + return pci_read_u16(.bus, .device, .fun, offset) + } + public fn read_u32(this, anon offset: i64) -> u32 { + return pci_read_u32(.bus, .device, .fun, offset) + } + public fn write_u8(this, offset: i64, value: u8) { + pci_write_u8(.bus, .device, .fun, offset, value) + } + public fn write_u16(this, offset: i64, value: u16) { + pci_write_u16(.bus, .device, .fun, offset, value) + } + public fn write_u32(this, offset: i64, value: u32) { + pci_write_u32(.bus, .device, .fun, offset, value) + } + public fn io_read_u8(this, anon offset: u16) throws -> u8 { + return IOPort::read_u8(.bar[0] as! u16 + offset) + } + public fn io_read_u16(this, anon offset: u16) throws -> u16 { + return IOPort::read_u16(.bar[0] as! u16 + offset) + } + public fn io_read_u32(this, anon offset: u16) throws -> u32 { + return IOPort::read_u32(.bar[0] as! u16 + offset) + } + public fn io_write_u8(this, offset: u16, value: u8) { + IOPort::write_u8(address: .bar[0] as! u16 + offset, value) + } + public fn io_write_u16(this, offset: u16, value: u16) { + IOPort::write_u16(address: .bar[0] as! u16 + offset, value) + } + public fn io_write_u32(this, offset: u16, value: u32) { + IOPort::write_u32(address: .bar[0] as! u16 + offset, value) + } + public fn vendor_id(this) -> u16 { + return .pci_vendor_id + } + public fn device_id(this) -> u16 { + return .pci_device_id + } + public fn command(this) -> u16 { + return pci_read_u16(.bus, .device, .fun, 0x4) + } + public fn set_command(this, anon value: u16) { + pci_write_u16(.bus, .device, .fun, offset: 0x4, value) + } + public fn status(this) -> u16 { + return pci_read_u16(.bus, .device, .fun, 0x6) + } +} + +fn lookup_bar(bus: i64, device: i64, fun: i64, anon index: i64) -> u32 { + if index < 0 or index > 5 { + return 0xFFFFFFFF + } + return pci_read_u32(bus, device, fun, 0x10 + (index * 4)) & 0xFFFFFFFC +} + +struct PCI { + public fn find_device_by_class_code(anon class_code: i64) throws -> PCIDevice { + let result = pci_find(class_code) + + if result < 0 { + eprintln("error: device not found") + throw Error::from_errno(1) + } + + let bus = (result >> 16) & 0xff + let device = (result >> 8) & 0xff + let fun = result & 0xff + let pci_vendor_id = pci_read_u16(bus, device, fun, 0x0) + let pci_device_id = pci_read_u16(bus, device, fun, 0x2) + mut bar: [u32] = [] + for i in 0..5 { + bar.push(lookup_bar(bus, device, fun, i)) + } + return PCIDevice(bus, device, fun, pci_vendor_id, pci_device_id, bar) + } +} diff --git a/src/net/os/time.jakt b/src/net/os/time.jakt new file mode 100644 index 0000000..c84daaa --- /dev/null +++ b/src/net/os/time.jakt @@ -0,0 +1,94 @@ +import extern c "time.h" { + extern fn time_busy(anon duration: i64) + extern fn time_jiffies() -> i64 + extern fn time_now() -> i64 + extern fn time_sleep(anon duration: i64) +} + +struct Time { + fn busy(anon duration: i64) { + time_busy(duration) + } + fn jiffies() throws -> i64 { + return time_jiffies() + } + fn now() throws -> i64 { + return time_now() + } + fn cdate_to_unix(anon cdate: i64) -> i64 { + // (cdate - Str2Date("1/1/1970") / CDATE_FREQ + NIST_TIME_OFFSET + return (cdate - 3090344933588992) / 49710 + 8575 + } + fn unix_to_cdate(anon unix: i64) -> i64 { + // (unix - NIST_TIME_OFFSET) * CDATE_FREQ + Str2Date("1/1/1970") + return (unix - 8575) * 49710 + 3090344933588992 + } + fn sleep(anon duration: i64) { + time_sleep(duration) + } + fn timestamp_from_unix(anon timestamp: i64) -> String { + + let SECS_PER_DAY = 86400 + let DAYS_PER_YEAR = 365 + let DAYS_PER_LYEAR = 366 + let DAYS_PER_LYEAR_PERIOD = 146097 + let YEARS_PER_LYEAR_PERIOD = 400 + + mut days = timestamp / SECS_PER_DAY + mut remainder = timestamp - (days * SECS_PER_DAY) + if timestamp < 0 and remainder == 0 { + days++ + remainder -= SECS_PER_DAY + } + + mut cur_year = 0 + mut months: [i64] = [] + mut tmp_days = 0 + + let month_tab = [ 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 ] + let month_tab_leap = [ -1, 30, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 ] + + tmp_days = days; + if tmp_days >= DAYS_PER_LYEAR_PERIOD or tmp_days <= -DAYS_PER_LYEAR_PERIOD { + cur_year += YEARS_PER_LYEAR_PERIOD * (tmp_days / DAYS_PER_LYEAR_PERIOD); + tmp_days -= DAYS_PER_LYEAR_PERIOD * (tmp_days / DAYS_PER_LYEAR_PERIOD); + } + while tmp_days >= DAYS_PER_LYEAR { + cur_year++; + if cur_year % 4 == 0 { + tmp_days -= DAYS_PER_LYEAR; + } else { + tmp_days -= DAYS_PER_YEAR; + } + } + if cur_year % 4 == 0 { + months = month_tab_leap + } else { + months = month_tab + } + + mut i = 11 + while i > 0 { + if tmp_days > months[i] { + break; + } + i-- + } + + let year = 1970 + cur_year + let month = i + 1 + let day = tmp_days - months[i] + + let hours = remainder / 3600 + let minutes = (remainder - hours * 3600) / 60 + let seconds = remainder % 60 + + mut sb = StringBuilder::create() + sb.clear() + sb.appendff("{:0>4d}-{:0>2d}-{:0>2d}T{:0>2d}:{:0>2d}:{:0>2d}.000Z", year, month, day, hours, minutes, seconds) + return sb.to_string() + } + fn timestamp_from_cdate(anon cdate: i64) -> String { + return timestamp_from_unix(cdate_to_unix(cdate)) + } +} \ No newline at end of file diff --git a/src/net/tcpip.jakt b/src/net/tcpip.jakt new file mode 100644 index 0000000..af5bd76 --- /dev/null +++ b/src/net/tcpip.jakt @@ -0,0 +1,1675 @@ +import lib::util { Util } + +import os::os { OS } +import os::time { Time } + +struct IcmpRequest { + host: u64 + identifier: u64 + sequence_number: u64 + response_type_pointer: u64 +} + +enum TcpSessionState: i64 { + Idle = 0 + Established = 1 + Closed = 2 + Connecting = 4 +} + +struct TcpSession { + local_mac: [u8] + remote_mac: [u8] + local_address: [u8] + remote_address: [u8] + local_port: u16 + remote_port: u16 + local_sequence_number: u32 + acknowledgement_number: u32 + timestamp_last_echo_reply: u32 + timestamp_last_jiffies: i64 + timestamp_origin: i64 + last_window_size: u16 + last_tx_sequence_number: u32 + last_tx_was_acked: bool + tx_chunk_counter: i64 + state: TcpSessionState + pending_data_to_transmit: [u8] + received_data: [u8] + received_frames: [[u8]] + is_bound_socket: bool + socket: u64 +} + +enum TcpBindError: u64 { + BadRequest = 1 + SocketIsAlreadyBound = 2 +} + +class TCPIP { + public ipv4_address: [u8] + public ipv4_netmask: [u8] + public ipv4_network: [u8] + public ipv4_gateway: [u8] + public dns_server_address: [u8] + public dns_server_port: u16 + public mss_size: u16 + public tx_queue: [[u8]] + public ttl: u8 + public arp_cache: [String:[u8]] + public bound_sockets: [u16:u64] + public dns_cache: [String:[u8]] + public tcp_sessions: [TcpSession] + public pending_dns_lookups: [u16:u64] + public pending_dns_cached_entries: [u16:String] + public pending_icmp_requests: [u16:u64] + public timestamp_last_arp_request: i64 + public rx_bytes: u64 + public rx_frames: u64 + public tx_bytes: u64 + public tx_frames: u64 + fn calculate_header_checksum(this, anon mut packet: [u8], offset: i64, count: i64) -> u16 { + mut sum: i64 = 0 + for i in 0..count { + if (i & 1) == 1 { + sum += packet[offset + i] as! i64 + } else { + sum += (packet[offset + i] as! i64 * 256) + } + } + mut calculated_checksum: u16 = 0 + let overflowable_checksum: i64 = (sum & 0xffff) + (sum >> 16) + if overflowable_checksum < 65536 { + calculated_checksum = ~((sum & 0xffff) as! u16 + (sum >> 16) as! u16) + } else { + calculated_checksum = ~(unchecked_add ((sum & 0xffff) as! u16, (sum >> 16) as! u16)) - 1 + } + return calculated_checksum + } + fn push_ethernet_header(this, mut packet: [u8], destination_mac: [u8], source_mac: [u8], ethertype: u16) throws { + for i in 0..6 { + packet.push(destination_mac[i]) + } + for i in 0..6 { + packet.push(source_mac[i]) + } + Util::push_u16_to_u8_array(packet, ethertype) + } + fn push_arp_packet(this, mut packet: [u8], opcode: u16, sender_mac: [u8], sender_ipv4: [u8], target_mac: [u8], target_ipv4: [u8]) throws { + let hardware_protocol_header_data: [u8] = [0x00, 0x01, 0x08, 0x00, 0x06, 0x04] + for i in 0..6 { + packet.push(hardware_protocol_header_data[i]) + } + Util::push_u16_to_u8_array(packet, opcode) + for i in 0..6 { + packet.push(sender_mac[i]) + } + for i in 0..4 { + packet.push(sender_ipv4[i]) + } + for i in 0..6 { + packet.push(target_mac[i]) + } + for i in 0..4 { + packet.push(target_ipv4[i]) + } + } + fn push_ipv4_header(this, mut packet: [u8], total_length: u16, flags: u8, protocol: u8, source_address: [u8], destination_address: [u8]) throws { + packet.push(0x45u8) // version & header length + packet.push(0x00u8) // differentiated services field + Util::push_u16_to_u8_array(packet, total_length) // total length of entire packet + let identification: u64 = OS::random() & 0xffff + Util::push_u16_to_u8_array(packet, identification as! u16) + packet.push(flags) + packet.push(0x00u8) // fragment size (unused) + packet.push(.ttl) // time-to-live + packet.push(protocol) // protocol number + Util::push_u16_to_u8_array(packet, 0 as! u16) // Checksum placeholder + for i in 0..4 { + packet.push(source_address[i]) + } + for i in 0..4 { + packet.push(destination_address[i]) + } + let checksum = .calculate_header_checksum(packet, offset: 14, count: 20) + packet[24] = (checksum >> 8) as! u8 + packet[25] = (checksum & 0xff) as! u8 + } + fn push_udp_header(this, mut packet: [u8], source_port: u16, destination_port: u16, length: i64) throws { + Util::push_u16_to_u8_array(packet, source_port) + Util::push_u16_to_u8_array(packet, destination_port) + Util::push_u16_to_u8_array(packet, length as! u16) + Util::push_u16_to_u8_array(packet, 0 as! u16) // Checksum placeholder + } + fn send_arp_reply(mut this, anon mac_address: [u8], anon sender_mac_address: [u8], anon sender_ipv4_address: [u8]) throws { + mut packet: [u8] = [] + .push_ethernet_header(packet, destination_mac: sender_mac_address, source_mac: mac_address, ethertype: 0x0806) + .push_arp_packet( + packet + opcode: 0x0002 + sender_mac: mac_address + sender_ipv4: .ipv4_address + target_mac: sender_mac_address + target_ipv4: sender_ipv4_address + ) + .tx_queue.push(packet) + } + public fn send_arp_request(mut this, anon mac_address: [u8], anon target_ipv4_address: [u8]) throws { + if .timestamp_last_arp_request > 0 and (.timestamp_last_arp_request + 5000) > Time::jiffies() { + return + } + mut packet: [u8] = [] + .push_ethernet_header(packet, destination_mac: [0xff, 0xff, 0xff, 0xff, 0xff, 0xff], source_mac: mac_address, ethertype: 0x0806) + .push_arp_packet( + packet + opcode: 0x0001 + sender_mac: mac_address + sender_ipv4: .ipv4_address + target_mac: [0x00, 0x00, 0x00, 0x00, 0x00, 0x00] + target_ipv4: target_ipv4_address + ) + .tx_queue.push(packet) + .timestamp_last_arp_request = Time::jiffies() + } + fn insert_arp_cache_entry(mut this, ipv4_address: [u8], mac_address: [u8]) throws { + .arp_cache[Util::get_hexadecimal_string_from_ipv4_u8_array(ipv4_address)] = mac_address + } + public fn process_arp_packet(mut this, anon mac_address: [u8], anon frame: [u8]) throws { + let arp_packet = frame[14..] + + let hardware_type = Util::get_u16_from_u8_arrayslice(arp_packet, 0) + let protocol_type = Util::get_u16_from_u8_arrayslice(arp_packet, 2) + let hardware_size = arp_packet[4] + let protocol_size = arp_packet[5] + let opcode = Util::get_u16_from_u8_arrayslice(arp_packet, 6) + + let sender_mac_address: [u8] = [arp_packet[8], arp_packet[9], arp_packet[10], + arp_packet[11], arp_packet[12], arp_packet[13]] + let sender_ipv4_address: [u8] = [arp_packet[14], arp_packet[15], arp_packet[16], arp_packet[17]] + + let target_mac_address: [u8] = [arp_packet[18], arp_packet[19], arp_packet[20], + arp_packet[21], arp_packet[22], arp_packet[23]] + let target_ipv4_address: [u8] = [arp_packet[24], arp_packet[25], arp_packet[26], arp_packet[27]] + + // If any of these mismatch, the packet is invalid + if hardware_type != 0x0001 or protocol_type != 0x0800 or + hardware_size != 6 or protocol_size != 4 { + return + } + + // Add the sender to ARP cache + .insert_arp_cache_entry(ipv4_address: sender_ipv4_address, mac_address: sender_mac_address) + //print("Updated ARP cache entry for {}: ", sender_ipv4_address) + //for i in 0..6 { + // print("{:0>2x}", sender_mac_address[i]) + // if (i < 5) { + // print(":") + // } + //} + //println(" ") + + match opcode { + 1 => { + // ARP request, check if our ipv4 address matches + for i in 0..4 { + if .ipv4_address[i] != target_ipv4_address[i] { + // doesn't match, break + break + } + } + // matches, send ARP reply + .send_arp_reply(mac_address, sender_mac_address, sender_ipv4_address) + } + else => { + // unsupported opcode, ignore + } + } + } + public fn send_dns_query(mut this, anon mac_address: [u8], anon query: String, anon pointer_to_u32: u64) throws { + + mut cached_result: u32 = 0 + if .dns_cache.contains(query) { + cached_result += (.dns_cache[query][0] as! u32 * (256 * 256 * 256) as! u32) + cached_result += (.dns_cache[query][1] as! u32 * (256 * 256) as! u32) + cached_result += (.dns_cache[query][2] as! u32 * 256 as! u32) + cached_result += .dns_cache[query][3] as! u32 + unsafe { + cpp { + "u32 *r = (u32*)pointer_to_u32; + r[0] = cached_result;" + } + } + return + } + + mut packet: [u8] = [] + mut dns_packet: [u8] = [] + + let transaction_id: u64 = OS::random() & 0xffff + Util::push_u16_to_u8_array(dns_packet, transaction_id as! u16) + + .pending_dns_lookups.set(transaction_id as! u16, pointer_to_u32) + .pending_dns_cached_entries.set(transaction_id as! u16, query) + + let dns_header_data: [u8] = [0x01, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00] + dns_packet.push_values(&dns_header_data) + + let segments = query.split(c'.') + for segment in segments { + dns_packet.push(segment.length() as! u8) + //println("segment_length: {} bytes", segment.length()) + unsafe { + cpp { + "const char *raw_string = segment.characters(); + for (int i = 0; i < segment.length(); i++) + dns_packet.push(raw_string[i]);" + } + } + } + dns_packet.push(0x00u8) + + let dns_footer_data: [u8] = [0x00, 0x01, 0x00, 0x01] + dns_packet.push_values(&dns_footer_data) + + let source_port: u16 = (OS::random() & 0xffff) as! u16 + let destination_port: u16 = .dns_server_port + + let destination_mac = .arp_cache[Util::get_hexadecimal_string_from_ipv4_u8_array(.ipv4_gateway)] + .push_ethernet_header(packet, destination_mac, source_mac: mac_address, ethertype: 0x0800) + .push_ipv4_header(packet, total_length: 28 + dns_packet.size() as! u16, flags: 0x00, protocol: 0x11, source_address: .ipv4_address, destination_address: .dns_server_address) + .push_udp_header(packet, source_port, destination_port, length: 8 + dns_packet.size() as! i64) + + packet.push_values(&dns_packet) + + // Calculate UDP checksum + + mut checksum_packet: [u8] = [] + // Source and destination IP + for i in 26..34 { + checksum_packet.push(packet[i]) + } + // Protocol + checksum_packet.push(0x00u8) + checksum_packet.push(0x11u8) + // UDP Packet length + let udp_packet_length: u16 = 8 + dns_packet.size() as! u16 + Util::push_u16_to_u8_array(checksum_packet, udp_packet_length) + // Source Port + Util::push_u16_to_u8_array(checksum_packet, source_port) + // Destination port + Util::push_u16_to_u8_array(checksum_packet, destination_port) + // UDP Packet length (again) + Util::push_u16_to_u8_array(checksum_packet, udp_packet_length) + // Data + checksum_packet.push_values(&dns_packet) + + let checksum = .calculate_header_checksum(checksum_packet, offset: 0, count: checksum_packet.size() as! i64) + packet[40] = (checksum >> 8) as! u8 + packet[41] = (checksum & 0xff) as! u8 + + .tx_queue.push(packet) + + } + public fn netinfo_process_client_request(mut this, anon mac_address: [u8]) throws { + unsafe { + cpp { + "u64 *request = (u64*)0x300030; + if (*request) { + int i = 0; + u64 *r = (u64*)*request; + + for (i = 0; i < 6; i++) { + r[0] = (r[0] << 8) | mac_address[i]; + if (i < 4) { + r[1] = (r[1] << 8) | this->ipv4_address[i]; + r[2] = (r[2] << 8) | this->ipv4_netmask[i]; + r[3] = (r[3] << 8) | this->ipv4_network[i]; + r[4] = (r[4] << 8) | this->ipv4_gateway[i]; + r[5] = (r[5] << 8) | this->dns_server_address[i]; + } + } + + r[6] = this->dns_server_port; + + r[7] = this->rx_bytes; + r[8] = this->rx_frames; + r[9] = this->tx_bytes; + r[10] = this->tx_frames; + + u32 *p = (u32*)r[11]; + p[0] = 1; // pointer_to_u32 + + *request = 0; + }" + } + } + } + + public fn icmp_process_client_request(mut this, anon mac_address: [u8]) throws { + mut did_receive_request = false + mut destination_mac: [u8] = [] + mut addr: u32 = 0 + mut iden: u64 = 0 + mut seq: u64 = 0 + mut pointer_to_u32: u64 = 0 + unsafe { + cpp { + "u64 *ls_request = (u64*)0x300020; + if (*ls_request) { + u64 *ls_r = (u64*)*ls_request; + addr = ls_r[0]; + }" + } + } + // FIXME: This will crash if we 1) Ping a host on the local subnet that does not exist, then 2) subsequently ping the gateway. + if (addr > 0) { + mut remote_address_hex_string = Util::get_hexadecimal_string_from_ipv4_u8_array(.ipv4_gateway) + if addr != Util::get_address_u32_from_ipv4_u8_array(.ipv4_gateway) { + if .is_local_ipv4_address(Util::get_ipv4_u8_array_from_address_u32(addr)) { + remote_address_hex_string = Util::get_hexadecimal_string_from_ipv4_u8_array(Util::get_ipv4_u8_array_from_address_u32(addr)) + if not .arp_cache.contains(remote_address_hex_string) { + // send arp request + //println("send arp request for: {}", remote_address_hex_string) + .send_arp_request(mac_address, Util::get_ipv4_u8_array_from_address_u32(addr)) + return + } + } + } + destination_mac = .arp_cache[remote_address_hex_string] + } else { + return + } + unsafe { + cpp { + "u64 *request = (u64*)0x300020; + if (*request) { + did_receive_request = true; + u64 *r = (u64*)*request; + addr = r[0]; + iden = r[1]; + seq = r[2]; + pointer_to_u32 = r[3]; + *request = 0; + }" + } + } + if (did_receive_request) { + .send_icmp_request(mac_address, destination_mac, addr, iden, seq, pointer_to_u32) + did_receive_request = false + } + } + + public fn dns_process_client_request(mut this, anon mac_address: [u8]) throws { + mut did_receive_request = false + mut request_is_ipv4_address = true + mut s = StringBuilder::create() + mut pointer_to_u32: u64 = 0 + unsafe { + cpp { + "u64 *request = (u64*)0x300010; + if (*request) { + did_receive_request = true; + u64 *r = (u64*)*request; + + char const *chars = (char const*)r[0]; + s.append_c_string(chars); + for (int i = 0; i < s.to_string().length(); i++) + if ((chars[i] < '0' || chars[i] > '9') && chars[i] != '.') + request_is_ipv4_address = false; + delete(chars); + + pointer_to_u32 = r[1]; + + *request = 0; + }" + } + } + if (did_receive_request) { + if (request_is_ipv4_address) { + let octets = s.to_string().split(c'.') + mut u32_address: u32 = 0 + u32_address += octets[3].to_number().value() + u32_address += octets[2].to_number().value() << 8 + u32_address += octets[1].to_number().value() << 16 + u32_address += octets[0].to_number().value() << 24 + unsafe { + cpp { + "u32 *r = (u32*)pointer_to_u32; + r[0] = u32_address;" + } + } + } else { + .send_dns_query(mac_address, s.to_string(), pointer_to_u32) + } + did_receive_request = false + } + } + fn send_icmp_request(mut this, anon mac_address: [u8], anon destination_mac: [u8], anon addr: u32, anon iden: u64, anon seq: u64, anon pointer_to_u32: u64) throws { + mut packet: [u8] = [] + + mut destination_address: [u8] = [] + Util::push_u32_to_u8_array(destination_address, addr) + + .pending_icmp_requests.set(iden as! u16, pointer_to_u32) + + let total_length: u16 = 84 + + .push_ethernet_header(packet, destination_mac, source_mac: mac_address, ethertype: 0x0800) + .push_ipv4_header(packet, total_length, flags: 0x40, protocol: 0x01, source_address: .ipv4_address, destination_address) + + packet.push(0x08u8) // type + for i in 0..3 { + packet.push(0x00u8) // code, checksum placeholder + } + + Util::push_u16_to_u8_array(packet, iden as! u16) + Util::push_u16_to_u8_array(packet, seq as! u16) + + for i in 0..8 { + packet.push(0x00u8) // FIXME: timestamp + } + + mut ch: u8 = 0x20 + for i in 0..48 { + packet.push(ch) // data + ch++ + } + + let checksum = .calculate_header_checksum(packet, offset: 34, count: packet.size() as! i64 - 34) + packet[36] = (checksum >> 8) as! u8 + packet[37] = (checksum & 0xff) as! u8 + + .tx_queue.push(packet) + } + fn process_icmp_reply(mut this, anon frame: [u8]) throws { + let iden = Util::get_u16_from_u8_array(frame, 38) + let ttl: u16 = frame[22] as! u16 + let payload_size: u16 = Util::get_u16_from_u8_array(frame, 16) - 20 + mut result: u32 = 0 + result = (payload_size as! u32) << 16 + result += ttl as! u32 + mut pointer_to_u32: u64 = 0 + if .pending_icmp_requests.contains(iden) { + pointer_to_u32 = .pending_icmp_requests[iden] + } + if pointer_to_u32 > 0 { + unsafe { + cpp { + "u32 *r = (u32*)pointer_to_u32; + r[0] = result;" + } + } + } + } + fn send_icmp_reply(mut this, anon mac_address: [u8], anon frame: [u8]) throws { + mut packet: [u8] = [] + let ipv4_packet = frame[14..] + let icmp_packet = frame[34..] + + mut destination_mac: [u8] = [] + for octet in frame[6..12] { + destination_mac.push(octet) + } + mut destination_address: [u8] = [] + for octet in ipv4_packet[12..16] { + destination_address.push(octet) + } + + .push_ethernet_header(packet, destination_mac, source_mac: mac_address, ethertype: 0x0800) + .push_ipv4_header(packet, total_length: frame.size() as! u16 - 14, flags: 0x40, protocol: 0x01, source_address: .ipv4_address, destination_address) + + for i in 0..4 { + packet.push(0x00u8) // type, code, checksum placeholder + } + for i in 4..8 { + packet.push(icmp_packet[i]) // identifier & sequence number + } + for i in 8..icmp_packet.size() { + packet.push(icmp_packet[i]) // data + } + let checksum = .calculate_header_checksum(packet, offset: 34, count: packet.size() as! i64 - 34) + packet[36] = (checksum >> 8) as! u8 + packet[37] = (checksum & 0xff) as! u8 + + .tx_queue.push(packet) + } + public fn process_icmp_packet(mut this, anon mac_address: [u8], anon frame: [u8]) throws { + let icmp_packet = frame[34..] + + let type = icmp_packet[0] + let code = icmp_packet[1] + let checksum: u16 = Util::get_u16_from_u8_arrayslice(icmp_packet, 2) + let identifier: u16 = Util::get_u16_from_u8_arrayslice(icmp_packet, 4) + let sequence_number: u16 = Util::get_u16_from_u8_arrayslice(icmp_packet, 6) + + match type { + 0 => { + // ICMP reply + .process_icmp_reply(frame) + } + 8 => { + // ICMP request, send ICMP reply + .send_icmp_reply(mac_address, frame) + } + else => { + // unsupported + } + } + } + fn tcp_is_fin(this, flags: u16) -> bool { + if (flags & 1) == 1 { + return true + } + return false + } + fn tcp_is_syn(this, flags: u16) -> bool { + if (flags & 2) == 2 { + return true + } + return false + } + fn tcp_is_reset(this, flags: u16) -> bool { + if (flags & 4) == 4 { + return true + } + return false + } + fn tcp_is_push(this, flags: u16) -> bool { + if (flags & 8) == 8 { + return true + } + return false + } + fn tcp_is_ack(this, flags: u16) -> bool { + if (flags & 16) == 16 { + return true + } + return false + } + fn tcp_is_urgent(this, flags: u16) -> bool { + if (flags & 32) == 32 { + return true + } + return false + } + fn tcp_syn_packet(mut this, anon mut session: TcpSession) throws { + mut packet: [u8] = [] + + let header_length: u16 = 40 + let flags: u16 = 0xa002 // SYN + + mut ipv4_total_length: u16 = 0 + ipv4_total_length += 20; // IPv4 + ipv4_total_length += header_length as! u16 // TCP + + .push_ethernet_header(packet, destination_mac: session.remote_mac, source_mac: session.local_mac, ethertype: 0x0800) + .push_ipv4_header(packet, total_length: ipv4_total_length, flags: 0x40, protocol: 0x06, source_address: session.local_address, destination_address: session.remote_address) + + // FIXME: put this into .push_tcp_header + + Util::push_u16_to_u8_array(packet, session.local_port) + Util::push_u16_to_u8_array(packet, session.remote_port) + Util::push_u32_to_u8_array(packet, session.local_sequence_number) + Util::push_u32_to_u8_array(packet, 0 as! u32) // Acknowledgement number + Util::push_u16_to_u8_array(packet, flags) + Util::push_u16_to_u8_array(packet, session.last_window_size) + Util::push_u16_to_u8_array(packet, 0 as! u16) // Checksum placeholder + Util::push_u16_to_u8_array(packet, 0 as! u16) // Urgent pointer + + mut tcp_options: [u8] = [0x02, 0x04] + Util::push_u16_to_u8_array(tcp_options, .mss_size) + Util::push_u32_to_u8_array(tcp_options, 0x0101080a) + packet.push_values(&tcp_options) + let timestamp: u32 = (session.timestamp_origin + (Time::jiffies() - session.timestamp_origin)) as! u32 + Util::push_u32_to_u8_array(packet, timestamp) + Util::push_u32_to_u8_array(packet, session.timestamp_last_echo_reply) + + packet.push(0x01u8) + packet.push(0x03u8) + packet.push(0x03u8) + packet.push(0x07u8) + + // Calculate TCP checksum + + mut checksum_packet: [u8] = [] + // Source and destination IP + for i in 26..34 { + checksum_packet.push(packet[i]) + } + // Protocol + checksum_packet.push(0x00u8) + checksum_packet.push(0x06u8) + // TCP Packet length + let tcp_packet_length: u16 = header_length as! u16 + Util::push_u16_to_u8_array(checksum_packet, tcp_packet_length) + + for i in 34..packet.size() { + checksum_packet.push(packet[i]) + } + + mut checksum = .calculate_header_checksum(checksum_packet, offset: 0, count: checksum_packet.size() as! i64) + packet[50] = (checksum >> 8) as! u8 + packet[51] = (checksum & 0xff) as! u8 + + .tx_queue.push(packet) + + } + fn tcp_psh_packet(mut this, anon mut session: TcpSession) throws { + mut packet: [u8] = [] + + let header_length: u16 = 32 + let flags: u16 = 0x8018 // PSH, ACK + + let maximum_payload_size_to_transmit: i64 = 1024 + mut payload_size: i64 = maximum_payload_size_to_transmit + let payload_offset: i64 = session.tx_chunk_counter * maximum_payload_size_to_transmit + if session.pending_data_to_transmit.size() as! i64 - payload_offset < maximum_payload_size_to_transmit { + payload_size = session.pending_data_to_transmit.size() as! i64 - payload_offset + } + + mut ipv4_total_length: u16 = 0 + ipv4_total_length += 20; // IPv4 + ipv4_total_length += header_length + payload_size as! u16 // TCP + + .push_ethernet_header(packet, destination_mac: session.remote_mac, source_mac: session.local_mac, ethertype: 0x0800) + .push_ipv4_header(packet, total_length: ipv4_total_length, flags: 0x40, protocol: 0x06, source_address: session.local_address, destination_address: session.remote_address) + + + // FIXME: put this into .push_tcp_header + + Util::push_u16_to_u8_array(packet, session.local_port) + Util::push_u16_to_u8_array(packet, session.remote_port) + Util::push_u32_to_u8_array(packet, session.local_sequence_number) + Util::push_u32_to_u8_array(packet, session.acknowledgement_number) + Util::push_u16_to_u8_array(packet, flags) + Util::push_u16_to_u8_array(packet, session.last_window_size) + Util::push_u16_to_u8_array(packet, 0 as! u16) // Checksum placeholder + Util::push_u16_to_u8_array(packet, 0 as! u16) // Urgent pointer + + mut tcp_options: [u8] = [0x01, 0x01, 0x08, 0x0a] + packet.push_values(&tcp_options) + + let timestamp: u32 = (session.timestamp_origin + (Time::jiffies() - session.timestamp_origin)) as! u32 + Util::push_u32_to_u8_array(packet, timestamp) + Util::push_u32_to_u8_array(packet, session.timestamp_last_echo_reply) + + for i in payload_offset..(payload_offset + payload_size) { + packet.push(session.pending_data_to_transmit[i]) + } + + // Calculate TCP checksum + + mut checksum_packet: [u8] = [] + // Source and destination IP + for i in 26..34 { + checksum_packet.push(packet[i]) + } + // Protocol + checksum_packet.push(0x00u8) + checksum_packet.push(0x06u8) + // TCP Packet length + let tcp_packet_length: u16 = header_length + payload_size as! u16 + Util::push_u16_to_u8_array(checksum_packet, tcp_packet_length) + + for i in 34..packet.size() { + checksum_packet.push(packet[i]) + } + + mut checksum = .calculate_header_checksum(checksum_packet, offset: 0, count: checksum_packet.size() as! i64) + packet[50] = (checksum >> 8) as! u8 + packet[51] = (checksum & 0xff) as! u8 + + .tx_queue.push(packet) + + } + public fn tcp_transmit_pending_data_for_existing_sessions(mut this) throws { + for i in 0..this.tcp_sessions.size() { + if .tcp_sessions[i].pending_data_to_transmit.size() > 0 { + if (.tcp_sessions[i].tx_chunk_counter == 0) or (.tcp_sessions[i].tx_chunk_counter > 0 and .tcp_sessions[i].last_tx_was_acked) { + let maximum_payload_size_to_transmit: i64 = 1024 + let payload_offset: i64 = .tcp_sessions[i].tx_chunk_counter * maximum_payload_size_to_transmit + mut truncated_packet: [u8] = [] + .tcp_psh_packet(.tcp_sessions[i]) + if (.tcp_sessions[i].pending_data_to_transmit.size() as! i64 - payload_offset >= maximum_payload_size_to_transmit) { + .tcp_sessions[i].last_tx_sequence_number = .tcp_sessions[i].local_sequence_number + maximum_payload_size_to_transmit as! u32 + .tcp_sessions[i].last_tx_was_acked = false + .tcp_sessions[i].tx_chunk_counter++ + } else { + // We have transmitted all of the pending data. + .tcp_sessions[i].pending_data_to_transmit.shrink(0) + .tcp_sessions[i].last_tx_sequence_number = 0 + .tcp_sessions[i].last_tx_was_acked = false + .tcp_sessions[i].tx_chunk_counter = 0 + } + } + } + } + } + fn tcp_ack_packet(mut this, anon mut session: TcpSession, anon frame: [u8], mut flags: u16) throws -> u32 { + mut packet: [u8] = [] + + let syn_ipv4_packet = frame[14..] + let syn_tcp_packet = frame[34..] + + // Swap source/destination ports + mut destination_port: u16 = Util::get_u16_from_u8_arrayslice(syn_tcp_packet, 0) + mut source_port: u16 = Util::get_u16_from_u8_arrayslice(syn_tcp_packet, 2) + + let header_length: u16 = ((syn_tcp_packet[12] as! u16) >> 4) * 4 + + flags += ((header_length / 4) << 12) + flags |= 16 // ACK + + mut sequence_number: u32 = session.local_sequence_number + mut acknowledgement_number: u32 = 0 + for i in 0..4 { + acknowledgement_number += (syn_tcp_packet[7 - i] as! u32) * match i { + 1 => 256 as! u32 + 2 => (256 * 256) as! u32 + 3 => (256 * 256 * 256) as! u32 + else => 1 as! u32 + } + } + + if (.tcp_is_syn(flags) or .tcp_is_fin(flags)) { + acknowledgement_number += 1 + } + if (.tcp_is_push(flags)) { + // advance ACK by number of bytes + flags -= 8 + } + acknowledgement_number += (syn_tcp_packet.size() as! u32 - header_length as! u32) + + let window_size: u16 = 0xffff + let urgent_pointer: u16 = Util::get_u16_from_u8_arrayslice(syn_tcp_packet, 18) + + mut destination_mac: [u8] = [] + for octet in frame[6..12] { + destination_mac.push(octet) + } + mut destination_address: [u8] = [] + for octet in syn_ipv4_packet[12..16] { + destination_address.push(octet) + } + + mut ipv4_total_length: u16 = 0 + ipv4_total_length += 20; // IPv4 + ipv4_total_length += header_length // TCP + + .push_ethernet_header(packet, destination_mac, source_mac: session.local_mac, ethertype: 0x0800) + .push_ipv4_header(packet, total_length: ipv4_total_length, flags: 0x40, protocol: 0x06, source_address: .ipv4_address, destination_address) + + // FIXME: put this into .push_tcp_header + + Util::push_u16_to_u8_array(packet, source_port) + Util::push_u16_to_u8_array(packet, destination_port) + Util::push_u32_to_u8_array(packet, sequence_number) + Util::push_u32_to_u8_array(packet, acknowledgement_number) + Util::push_u16_to_u8_array(packet, flags) + Util::push_u16_to_u8_array(packet, window_size) + Util::push_u16_to_u8_array(packet, 0 as! u16) // Checksum placeholder + Util::push_u16_to_u8_array(packet, urgent_pointer) + + mut tcp_options_offset = 20 + mut tcp_option_length = 0 + while tcp_options_offset < header_length as! i64 { + match syn_tcp_packet[tcp_options_offset] { + 0u8 => { + packet.push(0u8) + tcp_options_offset++ + } + 1u8 => { + packet.push(1u8) + tcp_options_offset++ + } + 8u8 => { + packet.push(8u8) + packet.push(10u8) + let timestamp: u32 = (session.timestamp_origin + (Time::jiffies() - session.timestamp_origin)) as! u32 + Util::push_u32_to_u8_array(packet, timestamp) + Util::push_u32_to_u8_array(packet, session.timestamp_last_echo_reply) + tcp_options_offset += 10 + } + else => { + packet.push(syn_tcp_packet[tcp_options_offset]) + packet.push(syn_tcp_packet[tcp_options_offset + 1]) + tcp_option_length = syn_tcp_packet[tcp_options_offset + 1] as! i64 + for i in 2..tcp_option_length { + packet.push(syn_tcp_packet[tcp_options_offset + i]) + } + tcp_options_offset += tcp_option_length + } + } + } + + // Calculate TCP checksum + + mut checksum_packet: [u8] = [] + // Source and destination IP + for i in 26..34 { + checksum_packet.push(packet[i]) + } + // Protocol + checksum_packet.push(0x00u8) + checksum_packet.push(0x06u8) + // TCP Packet length + let tcp_packet_length: u16 = 40 as! u16 + Util::push_u16_to_u8_array(checksum_packet, tcp_packet_length) + + for i in 34..packet.size() { + checksum_packet.push(packet[i]) + } + + mut checksum = .calculate_header_checksum(checksum_packet, offset: 0, count: checksum_packet.size() as! i64) + if (not .tcp_is_syn(flags)) { + checksum += 8 + } + packet[50] = (checksum >> 8) as! u8 + packet[51] = (checksum & 0xff) as! u8 + + .tx_queue.push(packet) + return acknowledgement_number + + } + fn tcp_synack_ack_packet(mut this, anon mut session: TcpSession, anon frame: [u8], mut flags: u16) throws -> u32 { + mut packet: [u8] = [] + + let syn_ipv4_packet = frame[14..] + let syn_tcp_packet = frame[34..] + + // Swap source/destination ports + mut destination_port: u16 = Util::get_u16_from_u8_arrayslice(syn_tcp_packet, 0) + mut source_port: u16 = Util::get_u16_from_u8_arrayslice(syn_tcp_packet, 2) + + let header_length: u16 = (20 + 12) as! u16 + + flags += ((header_length / 4) << 12) + flags |= 16 // ACK + + mut sequence_number: u32 = session.local_sequence_number + mut acknowledgement_number: u32 = 0 + for i in 0..4 { + acknowledgement_number += (syn_tcp_packet[7 - i] as! u32) * match i { + 1 => 256 as! u32 + 2 => (256 * 256) as! u32 + 3 => (256 * 256 * 256) as! u32 + else => 1 as! u32 + } + } + + acknowledgement_number += 1 + + let window_size: u16 = 0xffff + let urgent_pointer: u16 = Util::get_u16_from_u8_arrayslice(syn_tcp_packet, 18) + + mut destination_mac: [u8] = [] + for octet in frame[6..12] { + destination_mac.push(octet) + } + mut destination_address: [u8] = [] + for octet in syn_ipv4_packet[12..16] { + destination_address.push(octet) + } + + mut ipv4_total_length: u16 = 0 + ipv4_total_length += 20; // IPv4 + ipv4_total_length += header_length // TCP + + .push_ethernet_header(packet, destination_mac, source_mac: session.local_mac, ethertype: 0x0800) + .push_ipv4_header(packet, total_length: ipv4_total_length, flags: 0x40, protocol: 0x06, source_address: .ipv4_address, destination_address) + + // FIXME: put this into .push_tcp_header + + Util::push_u16_to_u8_array(packet, source_port) + Util::push_u16_to_u8_array(packet, destination_port) + Util::push_u32_to_u8_array(packet, sequence_number) + Util::push_u32_to_u8_array(packet, acknowledgement_number) + Util::push_u16_to_u8_array(packet, flags) + Util::push_u16_to_u8_array(packet, window_size) + Util::push_u16_to_u8_array(packet, 0 as! u16) // Checksum placeholder + Util::push_u16_to_u8_array(packet, urgent_pointer) + + // Options: NOP, NOP, Timestamp + packet.push(0x01u8) + packet.push(0x01u8) + Util::push_u16_to_u8_array(packet, 0x080a) + let timestamp: u32 = (session.timestamp_origin + (Time::jiffies() - session.timestamp_origin)) as! u32 + Util::push_u32_to_u8_array(packet, timestamp) + Util::push_u32_to_u8_array(packet, session.timestamp_last_echo_reply) + + // Calculate TCP checksum + + mut checksum_packet: [u8] = [] + // Source and destination IP + for i in 26..34 { + checksum_packet.push(packet[i]) + } + // Protocol + checksum_packet.push(0x00u8) + checksum_packet.push(0x06u8) + // TCP Packet length + let tcp_packet_length: u16 = 40 as! u16 + Util::push_u16_to_u8_array(checksum_packet, tcp_packet_length) + + for i in 34..packet.size() { + checksum_packet.push(packet[i]) + } + + mut checksum = .calculate_header_checksum(checksum_packet, offset: 0, count: checksum_packet.size() as! i64) + checksum += 8 + packet[50] = (checksum >> 8) as! u8 + packet[51] = (checksum & 0xff) as! u8 + + .tx_queue.push(packet) + return acknowledgement_number + + } + fn tcp_session_matches_current_session(mut this, anon session: TcpSession, anon local_address: [u8], anon remote_address: [u8], anon local_port: u16, anon remote_port: u16) -> bool { + if session.state as! i64 == TcpSessionState::Closed as! i64 { + return false + } + if (session.local_port != local_port) or (session.remote_port != remote_port) { + return false + } + for i in 0..4 { + if (session.local_address[i] != local_address[i]) { + return false + } + if (session.remote_address[i] != remote_address[i]) { + return false + } + } + return true + } + fn tcp_send(mut this, anon mut session: TcpSession, anon data: [u8]) throws { + session.pending_data_to_transmit.push_values(&data) + } + public fn tcp_process_client_send_requests(mut this, anon mac_address: [u8]) throws { + for i in 0..this.tcp_sessions.size() { + let socket = .tcp_sessions[i].socket + if socket > 0 { + mut data: [u8] = [] + unsafe { + cpp { + "u64 *s = (u64*)socket; + if (s[10] == 1) { + u8 *buffer = (u8*)s[7]; + for (int i=0; i < s[9]; i++) { + data.push(buffer[i]); + } + s[10] = 0; + }" + } + } + if data.size() > 0 { + .tcp_send(.tcp_sessions[i], data) + } + } + } + } + public fn tcp_process_client_received_data(mut this) throws { + for i in 0..this.tcp_sessions.size() { + mut session = .tcp_sessions[i] + let socket = session.socket + mut length = session.received_data.size() + mut max_length: usize = 0 + // s[4] = 0; // receive_buffer_size + if (socket > 0) { + unsafe { + cpp { + "u64 *s = (u64*)socket; + max_length = s[4];" + } + } + } + // FIXME: Should the client be responsible for malloc()ing the receive buffer? + if length > 65536 { + length = 65536 + } + if (length > max_length) { + length = max_length + } + if (socket > 0) and (length == 0) and (.tcp_sessions[i].state as! i64 == TcpSessionState::Closed as! i64) { + .tcp_update_socket_session_state(.tcp_sessions[i].state, .tcp_sessions[i].socket) + mut client_is_not_ready_to_receive_data = 0 + unsafe { + cpp { + "u64 *s = (u64*)socket; + client_is_not_ready_to_receive_data = s[6];" + } + } + let is_bound_socket = .tcp_sessions[i].is_bound_socket + if (not is_bound_socket and client_is_not_ready_to_receive_data == 0) or (is_bound_socket) { + // really close the connection + unsafe { + cpp { + "u64 *s = (u64*)socket; + s[5] = 0; + s[6] = 1; + if (s[3] > 0) + free((u8*)s[3]); + if (s[7] > 0) + free((u8*)s[7]); + " + } + } + .tcp_sessions[i].socket = 0 + } + } + if socket > 0 and length > 0 { + mut client_is_not_ready_to_receive_data = 0 + unsafe { + cpp { + "u64 *s = (u64*)socket; + client_is_not_ready_to_receive_data = s[6];" + } + } + if (client_is_not_ready_to_receive_data == 0) { + unsafe { + cpp { + "u64 *s = (u64*)socket; + u8 *buffer = (u8*)s[3]; + for (int i=0; i < length; i++) { + buffer[i] = session.received_data[i]; + } + s[5] = length; + s[6] = 1;" + } + } + if (length == session.received_data.size() as! usize) { + session.received_data.shrink(0) + } else { + mut remaining_received_data: [u8] = [] + for j in length..session.received_data.size() { + remaining_received_data.push(session.received_data[j]) + } + session.received_data.shrink(0) + for j in 0..remaining_received_data.size() { + session.received_data.push(remaining_received_data[j]) + } + } + } + } + } + } + fn tcp_is_next_frame(mut this, index: i64, frame: [u8]) throws -> bool { + let tcp_packet = frame[34..] + mut sequence_number: u32 = 0 + for o in 0..4 { + sequence_number += (tcp_packet[7 - o] as! u32) * match o { + 1 => 256 as! u32 + 2 => (256 * 256) as! u32 + 3 => (256 * 256 * 256) as! u32 + else => 1 as! u32 + } + } + if (.tcp_sessions[index].acknowledgement_number == 0) { + // session not established yet, we're (hopefully) in a SYN/ACK + return true + } + if (sequence_number == .tcp_sessions[index].acknowledgement_number) { + return true + } + return false + } + fn tcp_handle_received_frame(mut this, anon i: i64, anon frame: [u8]) throws { + let ipv4_packet = frame[14..] + let tcp_packet = frame[34..] + + let total_length = Util::get_u16_from_u8_arrayslice(ipv4_packet, 2) + 14 + + let source_address: [u8] = [ipv4_packet[12], ipv4_packet[13], ipv4_packet[14], ipv4_packet[15]] + let destination_address: [u8] = [ipv4_packet[16], ipv4_packet[17], ipv4_packet[18], ipv4_packet[19]] + + mut source_port: u16 = Util::get_u16_from_u8_arrayslice(tcp_packet, 0) + mut destination_port: u16 = Util::get_u16_from_u8_arrayslice(tcp_packet, 2) + + let header_length: u16 = ((tcp_packet[12] as! u16) >> 4) * 4 + let flags: u16 = Util::get_u16_from_u8_arrayslice(tcp_packet, 12) & 0xfff + + mut echo_reply: u32 = 0 + mut tcp_options_offset = 54 + + while tcp_options_offset < frame.size() as! i64 { + match frame[tcp_options_offset] { + 0u8 => { tcp_options_offset++ } + 1u8 => { tcp_options_offset++ } + 8u8 => { + tcp_options_offset++ + echo_reply += (frame[tcp_options_offset + 1] as! u32 * (256 * 256 * 256) as! u32) + echo_reply += (frame[tcp_options_offset + 2] as! u32 * (256 * 256) as! u32) + echo_reply += (frame[tcp_options_offset + 3] as! u32 * 256 as! u32) + echo_reply += frame[tcp_options_offset + 4] as! u32 + break + } + else => { tcp_options_offset += frame[tcp_options_offset + 1] as! i64 } + } + } + + if (flags & 0xff) != 0x10 { + // Received a packet with flags other than just ACK + for i in 0..this.tcp_sessions.size() { + if (.tcp_session_matches_current_session(.tcp_sessions[i], destination_address, source_address, destination_port, source_port)) { + mut received_data: [u8] = [] + for j in frame[(34u16 + header_length)..total_length] { + received_data.push(j) + } + for j in 0..received_data.size() { + .tcp_sessions[i].received_data.push(received_data[j]) + } + if (.tcp_is_fin(flags)) { + // Received request to close connection + //println("Closing connection to IPv4 address: {}", source_address) + if (.tcp_is_push(flags)) { + .tcp_sessions[i].acknowledgement_number = .tcp_ack_packet(.tcp_sessions[i], frame, flags: 9u16) + } else { + .tcp_sessions[i].acknowledgement_number = .tcp_ack_packet(.tcp_sessions[i], frame, flags: 1u16) + } + .tcp_sessions[i].state = TcpSessionState::Closed + } else if (.tcp_is_syn(flags)) and (.tcp_is_ack(flags)) { + // Received SYN, ACK + .tcp_sessions[i].timestamp_last_echo_reply = echo_reply + .tcp_sessions[i].timestamp_last_jiffies = Time::jiffies() + .tcp_sessions[i].local_sequence_number++ + .tcp_sessions[i].acknowledgement_number = .tcp_synack_ack_packet(.tcp_sessions[i], frame, flags: 0u16) + .tcp_sessions[i].state = TcpSessionState::Established + .tcp_update_socket_session_state(.tcp_sessions[i].state, .tcp_sessions[i].socket) + } else { + .tcp_sessions[i].timestamp_last_echo_reply = echo_reply + .tcp_sessions[i].timestamp_last_jiffies = Time::jiffies() + mut acknowledgement_number: u32 = 0 + for o in 0..4 { + acknowledgement_number += (tcp_packet[11 - o] as! u32) * match o { + 1 => 256 as! u32 + 2 => (256 * 256) as! u32 + 3 => (256 * 256 * 256) as! u32 + else => 1 as! u32 + } + } + .tcp_sessions[i].local_sequence_number = acknowledgement_number + if (.tcp_is_push(flags)) { + .tcp_sessions[i].acknowledgement_number = .tcp_ack_packet(.tcp_sessions[i], frame, flags: 8u16) + } else { + .tcp_sessions[i].acknowledgement_number = .tcp_ack_packet(.tcp_sessions[i], frame, flags: 0u16) + } + } + } + } + } else if (flags & 0xff) == 0x10 { + // Received a packet with flags == just ACK + for i in 0..this.tcp_sessions.size() { + if (.tcp_session_matches_current_session(.tcp_sessions[i], destination_address, source_address, destination_port, source_port)) { + mut received_data: [u8] = [] + for j in frame[(34u16 + header_length)..total_length] { + received_data.push(j) + } + for j in 0..received_data.size() { + .tcp_sessions[i].received_data.push(received_data[j]) + } + mut sequence_number: u32 = 0 + for o in 0..4 { + sequence_number += (tcp_packet[7 - o] as! u32) * match o { + 1 => 256 as! u32 + 2 => (256 * 256) as! u32 + 3 => (256 * 256 * 256) as! u32 + else => 1 as! u32 + } + } + mut acknowledgement_number: u32 = 0 + for o in 0..4 { + acknowledgement_number += (tcp_packet[11 - o] as! u32) * match o { + 1 => 256 as! u32 + 2 => (256 * 256) as! u32 + 3 => (256 * 256 * 256) as! u32 + else => 1 as! u32 + } + } + let window_size: u16 = 0xffff + if acknowledgement_number == .tcp_sessions[i].last_tx_sequence_number { + .tcp_sessions[i].last_tx_was_acked = true + } + .tcp_sessions[i].local_sequence_number = acknowledgement_number + if received_data.size() > 0 { + .tcp_sessions[i].acknowledgement_number = .tcp_ack_packet(.tcp_sessions[i], frame, flags: 0u16) + } else { + .tcp_sessions[i].acknowledgement_number = sequence_number + (received_data.size() as! u32) + } + .tcp_sessions[i].last_window_size = window_size + .tcp_sessions[i].timestamp_last_echo_reply = echo_reply + .tcp_sessions[i].timestamp_last_jiffies = Time::jiffies() + } + } + } + } + fn tcp_handle_received_frames(mut this, anon index: i64) throws { + mut unhandled_frames: [[u8]] = [] + for frame in .tcp_sessions[index].received_frames { + if .tcp_is_next_frame(index, frame) { + .tcp_handle_received_frame(index, frame) + } else { + unhandled_frames.push(frame) + } + } + .tcp_sessions[index].received_frames.shrink(0) + for frame in unhandled_frames { + .tcp_sessions[index].received_frames.push(frame) + } + } + fn process_udp_packet(mut this, anon mac_address: [u8], anon frame: [u8]) throws { + let ipv4_packet = frame[14..] + let udp_packet = frame[34..] + + let source_address: [u8] = [ipv4_packet[12], ipv4_packet[13], ipv4_packet[14], ipv4_packet[15]] + let destination_address: [u8] = [ipv4_packet[16], ipv4_packet[17], ipv4_packet[18], ipv4_packet[19]] + + mut source_port: u16 = Util::get_u16_from_u8_arrayslice(udp_packet, 0) + mut destination_port: u16 = Util::get_u16_from_u8_arrayslice(udp_packet, 2) + + if (Util::get_hexadecimal_string_from_ipv4_u8_array(source_address) == Util::get_hexadecimal_string_from_ipv4_u8_array(.dns_server_address)) and (source_port == .dns_server_port) { + let transaction_id: u16 = Util::get_u16_from_u8_arrayslice(udp_packet, 8) + let flags: u16 = Util::get_u16_from_u8_arrayslice(udp_packet, 10) + // questions : 12-13 + // answer_rrs: 14-15 + // authority_rrs: 16-17 + // additional_rrs: 18-19 + + if .pending_dns_lookups.contains(transaction_id) { + mut result: u32 = 0 + mut result_type: u16 = 0 + // skip over query data to get answer + mut response_pos = 20 + while (udp_packet[response_pos] > 0) { + response_pos += 1 + udp_packet[response_pos] as! i64 + } + response_pos += 5 // skip over null byte, type, class + response_pos += 2 // skip over answer: name + + result_type = 256 * udp_packet[response_pos] as! u16 + result_type += udp_packet[response_pos + 1] as! u16 + + mut pointer_to_u32 = .pending_dns_lookups[transaction_id] + mut cached_result: [u8] = [] + + match result_type { + 1 => { + // A record + response_pos += 10 + result += (udp_packet[response_pos] as! u32 * (256 * 256 * 256) as! u32) + result += (udp_packet[response_pos + 1] as! u32 * (256 * 256) as! u32) + result += (udp_packet[response_pos + 2] as! u32 * 256 as! u32) + result += udp_packet[response_pos + 3] as! u32 + cached_result.push(udp_packet[response_pos]) + cached_result.push(udp_packet[response_pos + 1]) + cached_result.push(udp_packet[response_pos + 2]) + cached_result.push(udp_packet[response_pos + 3]) + .dns_cache[.pending_dns_cached_entries[transaction_id]] = cached_result + } + 5 => { + // CNAME record + response_pos += 8 // skip over answer: type, class, ttl + let data_length = (256 * udp_packet[response_pos] as! i64) + udp_packet[response_pos + 1] as! i64 + response_pos += 2 + let data_pos = response_pos + mut pointer_pos = 0 + + mut s = StringBuilder::create() + mut length = 0 + + while (response_pos < data_pos + data_length) { + match udp_packet[response_pos] { + 0xc0 => { + response_pos++ + pointer_pos = 8 + udp_packet[response_pos++] as! i64 + while (udp_packet[pointer_pos] > 0) { + length = udp_packet[pointer_pos] as! i64 + pointer_pos++ + for i in 0..length { + s.append(udp_packet[pointer_pos++]) + } + if udp_packet[pointer_pos] > 0 { + s.append('.') + } + } + } + else => { + length = udp_packet[response_pos++] as! i64 + for i in 0..length { + s.append(udp_packet[response_pos++]) + } + if udp_packet[response_pos] > 0 { + s.append('.') + } + } + } + } + .send_dns_query(mac_address, s.to_string(), pointer_to_u32) + pointer_to_u32 = 0 + } + else => { + // Unsupported record + response_pos += 10 // skip over answer: type, class, ttl, data length + result = 0xFFFFFFFF + } + } + if pointer_to_u32 > 0 { + unsafe { + cpp { + "u32 *r = (u32*)pointer_to_u32; + r[0] = result;" + } + } + } + .pending_dns_lookups.remove(transaction_id) + } + } + + } + fn is_listening_port(this, anon port: u16) -> bool { + return .bound_sockets.contains(port) + } + fn process_tcp_packet(mut this, anon mac_address: [u8], anon frame: [u8]) throws { + let ipv4_packet = frame[14..] + let tcp_packet = frame[34..] + + let source_address: [u8] = [ipv4_packet[12], ipv4_packet[13], ipv4_packet[14], ipv4_packet[15]] + let destination_address: [u8] = [ipv4_packet[16], ipv4_packet[17], ipv4_packet[18], ipv4_packet[19]] + + mut source_port: u16 = Util::get_u16_from_u8_arrayslice(tcp_packet, 0) + mut destination_port: u16 = Util::get_u16_from_u8_arrayslice(tcp_packet, 2) + + let header_length: u16 = ((tcp_packet[12] as! u16) >> 4) * 4 + let flags: u16 = Util::get_u16_from_u8_arrayslice(tcp_packet, 12) & 0xfff + + mut echo_reply: u32 = 0 + mut tcp_options_offset = 54 + + while tcp_options_offset < frame.size() as! i64 { + match frame[tcp_options_offset] { + 0u8 => { tcp_options_offset++ } + 1u8 => { tcp_options_offset++ } + 8u8 => { + tcp_options_offset++ + echo_reply += (frame[tcp_options_offset + 1] as! u32 * (256 * 256 * 256) as! u32) + echo_reply += (frame[tcp_options_offset + 2] as! u32 * (256 * 256) as! u32) + echo_reply += (frame[tcp_options_offset + 3] as! u32 * 256 as! u32) + echo_reply += frame[tcp_options_offset + 4] as! u32 + break + } + else => { tcp_options_offset += frame[tcp_options_offset + 1] as! i64 } + } + } + + if .tcp_is_syn(flags) and not (.tcp_is_ack(flags)) and (.is_listening_port(destination_port)) { + // Received SYN, begin a new connection + mut remote_mac: [u8] = [] + for i in 6..12 { + remote_mac.push(frame[i]) + } + if .is_local_ipv4_address(source_address) { + if not .arp_cache.contains(Util::get_hexadecimal_string_from_ipv4_u8_array(source_address)) { + // send arp request + //println("send arp request for: {}", Util::get_hexadecimal_string_from_ipv4_u8_array(source_address)) + .send_arp_request(mac_address, source_address) + } + } + mut socket: u64 = 0 + let function: u64 = .bound_sockets[destination_port] + let source_address_u32 = Util::get_address_u32_from_ipv4_u8_array(source_address) + unsafe { + cpp { + " + u64* s = (u64*)calloc(256, 1); + + s[0] = source_address_u32; + s[1] = source_port; + s[2] = 1; // TCP_SOCKET_STATE_ESTABLISHED + s[3] = (u64)calloc(65536, 1); // receive_buffer_ptr + s[4] = 0; // receive_buffer_size + s[5] = 0; // receive_buffer_filled + s[6] = 0; // receive_buffer_kick + s[7] = (u64)calloc(65536, 1); // send_buffer_ptr + s[8] = 65536; // send_buffer_size + s[9] = 0; // send_buffer_filled + s[10] = 0; // send_buffer_kick + + socket = (u64)s; + if (function && socket) { + os_call(function, socket); + } + " + } + } + mut session = TcpSession( + local_mac: mac_address + remote_mac: remote_mac + local_address: destination_address + remote_address: source_address + local_port: destination_port + remote_port: source_port + local_sequence_number: (OS::random() & 0xffffffff) as! u32 + acknowledgement_number: 0 + timestamp_last_echo_reply: 0 + timestamp_last_jiffies: 0 + timestamp_origin: Time::jiffies() + last_window_size: 0 + last_tx_sequence_number: 0 + last_tx_was_acked: false + tx_chunk_counter: 0 + state: TcpSessionState::Established + pending_data_to_transmit: [] + received_data: [] + received_frames: [] + is_bound_socket: true + socket) + //println("Accepting connection from IPv4 address: {}", source_address) + session.timestamp_last_echo_reply = echo_reply + session.timestamp_last_jiffies = Time::jiffies() + session.acknowledgement_number = .tcp_ack_packet(session, frame, flags) + .tcp_sessions.push(session) + } else { + // Add packet to list to be processed + for i in 0..this.tcp_sessions.size() { + if (.tcp_session_matches_current_session(.tcp_sessions[i], destination_address, source_address, destination_port, source_port)) { + .tcp_sessions[i].received_frames.push(frame) + .tcp_handle_received_frames(i as! i64) + } + } + } + } + public fn process_ipv4_packet(mut this, anon mac_address: [u8], anon frame: [u8]) throws { + let ipv4_packet = frame[14..] + + let version = ipv4_packet[0] >> 4 + let header_length = (ipv4_packet[0] as! i64 & 0xf) * 4 + let dsf = ipv4_packet[1] + let total_length = Util::get_u16_from_u8_arrayslice(ipv4_packet, 2) + let identification = Util::get_u16_from_u8_arrayslice(ipv4_packet, 4) + let flags = ipv4_packet[6] + let ttl = ipv4_packet[8] + let protocol = ipv4_packet[9] + let checksum = Util::get_u16_from_u8_arrayslice(ipv4_packet, 10) + let source_address: [u8] = [ipv4_packet[12], ipv4_packet[13], ipv4_packet[14], ipv4_packet[15]] + let destination_address: [u8] = [ipv4_packet[16], ipv4_packet[17], ipv4_packet[18], ipv4_packet[19]] + + match protocol { + 1 => { + .process_icmp_packet(mac_address, frame) + } + 6 => { + .process_tcp_packet(mac_address, frame) + } + 17 => { + .process_udp_packet(mac_address, frame) + } + else => { + // unsupported + } + } + } + public fn set_ipv4_address(mut this, anon ipv4_address: [i64]) { + for i in 0..4 { + .ipv4_address[i] = ipv4_address[i] as! u8 + } + } + public fn set_ipv4_netmask(mut this, anon ipv4_netmask: [i64]) { + for i in 0..4 { + .ipv4_netmask[i] = ipv4_netmask[i] as! u8 + } + } + public fn set_ipv4_network(mut this, anon ipv4_network: [i64]) { + for i in 0..4 { + .ipv4_network[i] = ipv4_network[i] as! u8 + } + } + public fn set_ipv4_gateway(mut this, anon ipv4_gateway: [i64]) { + for i in 0..4 { + .ipv4_gateway[i] = ipv4_gateway[i] as! u8 + } + } + fn is_local_ipv4_address(this, anon check_ipv4_address: [u8]) -> bool { + let check_ipv4_address_u32 = Util::get_address_u32_from_ipv4_u8_array(check_ipv4_address) + let local_network_address_u32 = Util::get_address_u32_from_ipv4_u8_array(.ipv4_network) + let local_netmask_address_u32 = Util::get_address_u32_from_ipv4_u8_array(.ipv4_netmask) + // NOTE: The line below was added and the subsequent line was modified after jakt PR #1596 to suppress a warning from codegen output + // tcpip.cpp:2599:90: warning: & has lower precedence than ==; == will be evaluated first [-Wparentheses] + // 2599 | return (local_network_address_u32 & local_netmask_address_u32) == check_ipv4_address_u32 & local_netmask_address_u32; + // | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~^ + // tcpip.cpp:2599:90: note: place parentheses around the '==' expression to silence this warning + let check_ipv4_address_u32_and_with_netmask = check_ipv4_address_u32 & local_netmask_address_u32 + return (local_network_address_u32 & local_netmask_address_u32) == check_ipv4_address_u32_and_with_netmask + } + fn tcp_update_socket_session_state(mut this, anon state: TcpSessionState, anon socket: u64) throws { + unsafe { + cpp { + "u64 *s = (u64*)socket; + s[2] = (u64)state; + " + } + } + } + public fn tcp_process_bind_request(mut this) { + mut did_receive_request = false + mut port: u16 = 0 + mut function: u64 = 0 + mut response_code: u64 = 0 + unsafe { + cpp { + " + u64 *request = (u64*)0x300040; + if (*request) { + did_receive_request = true; + u64 *b = (u64*)*request; + port = b[0]; + function = b[1]; + } + " + } + } + if did_receive_request { + if .bound_sockets.contains(port) { + response_code = TcpBindError::SocketIsAlreadyBound as! u64 + } else if port < 1 or function < 1 { + response_code = TcpBindError::BadRequest as! u64 + } else { + .bound_sockets[port] = function + } + unsafe { + cpp { + " + u64 *request = (u64*)0x300040; + u64 *b = (u64*)*request; + b[2] = response_code; + *request = 0; + " + } + } + } + } + public fn tcp_process_client_socket_request(mut this, anon mac_address: [u8]) throws { + mut did_receive_request = false + mut remote_address: u32 = 0 + mut remote_port: u16 = 0 + mut session_remote_mac: [u8] = [] + mut socket: u64 = 0 + unsafe { + cpp { + "u64 *ls_request = (u64*)0x300000; + if (*ls_request) { + u64 *ls_s = (u64*)*ls_request; + remote_address = ls_s[0]; + }" + } + } + if (remote_address > 0) { + mut remote_address_hex_string = Util::get_hexadecimal_string_from_ipv4_u8_array(.ipv4_gateway) + if .is_local_ipv4_address(Util::get_ipv4_u8_array_from_address_u32(remote_address)) { + remote_address_hex_string = Util::get_hexadecimal_string_from_ipv4_u8_array(Util::get_ipv4_u8_array_from_address_u32(remote_address)) + if not .arp_cache.contains(remote_address_hex_string) { + // send arp request + //println("send arp request for: {}", remote_address_hex_string) + .send_arp_request(mac_address, Util::get_ipv4_u8_array_from_address_u32(remote_address)) + return + } + } + session_remote_mac = .arp_cache[remote_address_hex_string] + } else { + return + } + unsafe { + cpp { + "u64 *request = (u64*)0x300000; + if (*request) { + did_receive_request = true; + u64 *s = (u64*)*request; + + s[2] = 5; // TCP_SOCKET_STATE_IDLE + + s[3] = (u64)calloc(65536, 1); // receive_buffer_ptr + s[4] = 0; // receive_buffer_size + s[5] = 0; // receive_buffer_filled + s[6] = 1; // receive_buffer_kick + + s[7] = (u64)calloc(65536, 1); // send_buffer_ptr + s[8] = 65536; // send_buffer_size + s[9] = 0; // send_buffer_filled + s[10] = 0; // send_buffer_kick + + remote_address = s[0]; + remote_port = s[1]; + socket = (u64)s; + *request = 0; + }" + } + } + if (did_receive_request) { + let local_sequence_number: u32 = (OS::random() & 0xffffffff) as! u32 + + mut session = TcpSession( + local_mac: mac_address + remote_mac: session_remote_mac + local_address: .ipv4_address + remote_address: Util::get_ipv4_u8_array_from_address_u32(remote_address) + local_port: (OS::random() & 0xffff) as! u16 // FIXME + remote_port + local_sequence_number + acknowledgement_number: 0 + timestamp_last_echo_reply: 0 + timestamp_last_jiffies: Time::jiffies() + timestamp_origin: Time::jiffies() + last_window_size: 0xffff + last_tx_sequence_number: 0 + last_tx_was_acked: false + tx_chunk_counter: 0 + state: TcpSessionState::Connecting + pending_data_to_transmit: [] + received_data: [] + received_frames: [] + is_bound_socket: false + socket) + .tcp_syn_packet(session) + .tcp_update_socket_session_state(session.state, session.socket) + .tcp_sessions.push(session) + did_receive_request = false + } + } +} \ No newline at end of file diff --git a/src/tlse/LICENSE b/src/tlse/LICENSE new file mode 100644 index 0000000..6494b3d --- /dev/null +++ b/src/tlse/LICENSE @@ -0,0 +1,60 @@ +This software is available under your choice of one of the following licenses. +Choose whichever you prefer. + +=============================================================================== +License Choice 1 - The 2-Clause BSD License +=============================================================================== + +Copyright (c) 2016 - 2024, Eduard Suica +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + +* Redistributions of source code must retain the above copyright notice, this + list of conditions and the following disclaimer. + +* Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +=============================================================================== +License Choice 2 - Public Domain (Unlicense) +=============================================================================== + +This is free and unencumbered software released into the public domain. + +Anyone is free to copy, modify, publish, use, compile, sell, or +distribute this software, either in source code form or as a compiled +binary, for any purpose, commercial or non-commercial, and by any +means. + +In jurisdictions that recognize copyright laws, the author or authors +of this software dedicate any and all copyright interest in the +software to the public domain. We make this dedication for the benefit +of the public at large and to the detriment of our heirs and +successors. We intend this dedication to be an overt act of +relinquishment in perpetuity of all present and future rights to this +software under copyright law. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. +IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR +OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, +ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. + +For more information, please refer to + diff --git a/src/tlse/libtomcrypt.c b/src/tlse/libtomcrypt.c new file mode 100644 index 0000000..b6e9c33 --- /dev/null +++ b/src/tlse/libtomcrypt.c @@ -0,0 +1,34766 @@ +#define CRYPT 0x0117 +#define LTC_NO_ROLC + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://math.libtomcrypt.com + */ +#ifndef BN_H_ +#define BN_H_ + +#include +#include +#include +#include + +#if !(defined(LTM1) && defined(LTM2) && defined(LTM3)) + #if defined(LTM2) + #define LTM3 + #endif + #if defined(LTM1) + #define LTM2 + #endif + #define LTM1 + + #if defined(LTM_ALL) + #define BN_ERROR_C + #define BN_FAST_MP_INVMOD_C + #define BN_FAST_MP_MONTGOMERY_REDUCE_C + #define BN_FAST_S_MP_MUL_DIGS_C + #define BN_FAST_S_MP_MUL_HIGH_DIGS_C + #define BN_FAST_S_MP_SQR_C + #define BN_MP_2EXPT_C + #define BN_MP_ABS_C + #define BN_MP_ADD_C + #define BN_MP_ADD_D_C + #define BN_MP_ADDMOD_C + #define BN_MP_AND_C + #define BN_MP_CLAMP_C + #define BN_MP_CLEAR_C + #define BN_MP_CLEAR_MULTI_C + #define BN_MP_CMP_C + #define BN_MP_CMP_D_C + #define BN_MP_CMP_MAG_C + #define BN_MP_CNT_LSB_C + #define BN_MP_COPY_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_DIV_C + #define BN_MP_DIV_2_C + #define BN_MP_DIV_2D_C + #define BN_MP_DIV_3_C + #define BN_MP_DIV_D_C + #define BN_MP_DR_IS_MODULUS_C + #define BN_MP_DR_REDUCE_C + #define BN_MP_DR_SETUP_C + #define BN_MP_EXCH_C + #define BN_MP_EXPORT_C + #define BN_MP_EXPT_D_C + #define BN_MP_EXPT_D_EX_C + #define BN_MP_EXPTMOD_C + #define BN_MP_EXPTMOD_FAST_C + #define BN_MP_EXTEUCLID_C + #define BN_MP_FREAD_C + #define BN_MP_FWRITE_C + #define BN_MP_GCD_C + #define BN_MP_GET_INT_C + #define BN_MP_GET_LONG_C + #define BN_MP_GET_LONG_LONG_C + #define BN_MP_GROW_C + #define BN_MP_IMPORT_C + #define BN_MP_INIT_C + #define BN_MP_INIT_COPY_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_INIT_SET_C + #define BN_MP_INIT_SET_INT_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_INVMOD_C + #define BN_MP_INVMOD_SLOW_C + #define BN_MP_IS_SQUARE_C + #define BN_MP_JACOBI_C + #define BN_MP_KARATSUBA_MUL_C + #define BN_MP_KARATSUBA_SQR_C + #define BN_MP_LCM_C + #define BN_MP_LSHD_C + #define BN_MP_MOD_C + #define BN_MP_MOD_2D_C + #define BN_MP_MOD_D_C + #define BN_MP_MONTGOMERY_CALC_NORMALIZATION_C + #define BN_MP_MONTGOMERY_REDUCE_C + #define BN_MP_MONTGOMERY_SETUP_C + #define BN_MP_MUL_C + #define BN_MP_MUL_2_C + #define BN_MP_MUL_2D_C + #define BN_MP_MUL_D_C + #define BN_MP_MULMOD_C + #define BN_MP_N_ROOT_C + #define BN_MP_N_ROOT_EX_C + #define BN_MP_NEG_C + #define BN_MP_OR_C + #define BN_MP_PRIME_FERMAT_C + #define BN_MP_PRIME_IS_DIVISIBLE_C + #define BN_MP_PRIME_IS_PRIME_C + #define BN_MP_PRIME_MILLER_RABIN_C + #define BN_MP_PRIME_NEXT_PRIME_C + #define BN_MP_PRIME_RABIN_MILLER_TRIALS_C + #define BN_MP_PRIME_RANDOM_EX_C + #define BN_MP_RADIX_SIZE_C + #define BN_MP_RADIX_SMAP_C + #define BN_MP_RAND_C + #define BN_MP_READ_RADIX_C + #define BN_MP_READ_SIGNED_BIN_C + #define BN_MP_READ_UNSIGNED_BIN_C + #define BN_MP_REDUCE_C + #define BN_MP_REDUCE_2K_C + #define BN_MP_REDUCE_2K_L_C + #define BN_MP_REDUCE_2K_SETUP_C + #define BN_MP_REDUCE_2K_SETUP_L_C + #define BN_MP_REDUCE_IS_2K_C + #define BN_MP_REDUCE_IS_2K_L_C + #define BN_MP_REDUCE_SETUP_C + #define BN_MP_RSHD_C + #define BN_MP_SET_C + #define BN_MP_SET_INT_C + #define BN_MP_SET_LONG_C + #define BN_MP_SET_LONG_LONG_C + #define BN_MP_SHRINK_C + #define BN_MP_SIGNED_BIN_SIZE_C + #define BN_MP_SQR_C + #define BN_MP_SQRMOD_C + #define BN_MP_SQRT_C + #define BN_MP_SQRTMOD_PRIME_C + #define BN_MP_SUB_C + #define BN_MP_SUB_D_C + #define BN_MP_SUBMOD_C + #define BN_MP_TO_SIGNED_BIN_C + #define BN_MP_TO_SIGNED_BIN_N_C + #define BN_MP_TO_UNSIGNED_BIN_C + #define BN_MP_TO_UNSIGNED_BIN_N_C + #define BN_MP_TOOM_MUL_C + #define BN_MP_TOOM_SQR_C + #define BN_MP_TORADIX_C + #define BN_MP_TORADIX_N_C + #define BN_MP_UNSIGNED_BIN_SIZE_C + #define BN_MP_XOR_C + #define BN_MP_ZERO_C + #define BN_PRIME_TAB_C + #define BN_REVERSE_C + #define BN_S_MP_ADD_C + #define BN_S_MP_EXPTMOD_C + #define BN_S_MP_MUL_DIGS_C + #define BN_S_MP_MUL_HIGH_DIGS_C + #define BN_S_MP_SQR_C + #define BN_S_MP_SUB_C + #define BNCORE_C + #endif + + #if defined(BN_ERROR_C) + #define BN_MP_ERROR_TO_STRING_C + #endif + + #if defined(BN_FAST_MP_INVMOD_C) + #define BN_MP_ISEVEN_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_COPY_C + #define BN_MP_MOD_C + #define BN_MP_SET_C + #define BN_MP_DIV_2_C + #define BN_MP_ISODD_C + #define BN_MP_SUB_C + #define BN_MP_CMP_C + #define BN_MP_ISZERO_C + #define BN_MP_CMP_D_C + #define BN_MP_ADD_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_FAST_MP_MONTGOMERY_REDUCE_C) + #define BN_MP_GROW_C + #define BN_MP_RSHD_C + #define BN_MP_CLAMP_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_FAST_S_MP_MUL_DIGS_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_FAST_S_MP_MUL_HIGH_DIGS_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_FAST_S_MP_SQR_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_2EXPT_C) + #define BN_MP_ZERO_C + #define BN_MP_GROW_C + #endif + + #if defined(BN_MP_ABS_C) + #define BN_MP_COPY_C + #endif + + #if defined(BN_MP_ADD_C) + #define BN_S_MP_ADD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_ADD_D_C) + #define BN_MP_GROW_C + #define BN_MP_SUB_D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_ADDMOD_C) + #define BN_MP_INIT_C + #define BN_MP_ADD_C + #define BN_MP_CLEAR_C + #define BN_MP_MOD_C + #endif + + #if defined(BN_MP_AND_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_CLAMP_C) + #endif + + #if defined(BN_MP_CLEAR_C) + #endif + + #if defined(BN_MP_CLEAR_MULTI_C) + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_CMP_C) + #define BN_MP_CMP_MAG_C + #endif + + #if defined(BN_MP_CMP_D_C) + #endif + + #if defined(BN_MP_CMP_MAG_C) + #endif + + #if defined(BN_MP_CNT_LSB_C) + #define BN_MP_ISZERO_C + #endif + + #if defined(BN_MP_COPY_C) + #define BN_MP_GROW_C + #endif + + #if defined(BN_MP_COUNT_BITS_C) + #endif + + #if defined(BN_MP_DIV_C) + #define BN_MP_ISZERO_C + #define BN_MP_CMP_MAG_C + #define BN_MP_COPY_C + #define BN_MP_ZERO_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_SET_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_ABS_C + #define BN_MP_MUL_2D_C + #define BN_MP_CMP_C + #define BN_MP_SUB_C + #define BN_MP_ADD_C + #define BN_MP_DIV_2D_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_MULTI_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_INIT_C + #define BN_MP_INIT_COPY_C + #define BN_MP_LSHD_C + #define BN_MP_RSHD_C + #define BN_MP_MUL_D_C + #define BN_MP_CLAMP_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_DIV_2_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_DIV_2D_C) + #define BN_MP_COPY_C + #define BN_MP_ZERO_C + #define BN_MP_INIT_C + #define BN_MP_MOD_2D_C + #define BN_MP_CLEAR_C + #define BN_MP_RSHD_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #endif + + #if defined(BN_MP_DIV_3_C) + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_DIV_D_C) + #define BN_MP_ISZERO_C + #define BN_MP_COPY_C + #define BN_MP_DIV_2D_C + #define BN_MP_DIV_3_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_DR_IS_MODULUS_C) + #endif + + #if defined(BN_MP_DR_REDUCE_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_DR_SETUP_C) + #endif + + #if defined(BN_MP_EXCH_C) + #endif + + #if defined(BN_MP_EXPORT_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_DIV_2D_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_EXPT_D_C) + #define BN_MP_EXPT_D_EX_C + #endif + + #if defined(BN_MP_EXPT_D_EX_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_SET_C + #define BN_MP_MUL_C + #define BN_MP_CLEAR_C + #define BN_MP_SQR_C + #endif + + #if defined(BN_MP_EXPTMOD_C) + #define BN_MP_INIT_C + #define BN_MP_INVMOD_C + #define BN_MP_CLEAR_C + #define BN_MP_ABS_C + #define BN_MP_CLEAR_MULTI_C + #define BN_MP_REDUCE_IS_2K_L_C + #define BN_S_MP_EXPTMOD_C + #define BN_MP_DR_IS_MODULUS_C + #define BN_MP_REDUCE_IS_2K_C + #define BN_MP_ISODD_C + #define BN_MP_EXPTMOD_FAST_C + #endif + + #if defined(BN_MP_EXPTMOD_FAST_C) + #define BN_MP_COUNT_BITS_C + #define BN_MP_INIT_C + #define BN_MP_CLEAR_C + #define BN_MP_MONTGOMERY_SETUP_C + #define BN_FAST_MP_MONTGOMERY_REDUCE_C + #define BN_MP_MONTGOMERY_REDUCE_C + #define BN_MP_DR_SETUP_C + #define BN_MP_DR_REDUCE_C + #define BN_MP_REDUCE_2K_SETUP_C + #define BN_MP_REDUCE_2K_C + #define BN_MP_MONTGOMERY_CALC_NORMALIZATION_C + #define BN_MP_MULMOD_C + #define BN_MP_SET_C + #define BN_MP_MOD_C + #define BN_MP_COPY_C + #define BN_MP_SQR_C + #define BN_MP_MUL_C + #define BN_MP_EXCH_C + #endif + + #if defined(BN_MP_EXTEUCLID_C) + #define BN_MP_INIT_MULTI_C + #define BN_MP_SET_C + #define BN_MP_COPY_C + #define BN_MP_ISZERO_C + #define BN_MP_DIV_C + #define BN_MP_MUL_C + #define BN_MP_SUB_C + #define BN_MP_NEG_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_FREAD_C) + #define BN_MP_ZERO_C + #define BN_MP_S_RMAP_C + #define BN_MP_MUL_D_C + #define BN_MP_ADD_D_C + #define BN_MP_CMP_D_C + #endif + + #if defined(BN_MP_FWRITE_C) + #define BN_MP_RADIX_SIZE_C + #define BN_MP_TORADIX_C + #endif + + #if defined(BN_MP_GCD_C) + #define BN_MP_ISZERO_C + #define BN_MP_ABS_C + #define BN_MP_INIT_COPY_C + #define BN_MP_CNT_LSB_C + #define BN_MP_DIV_2D_C + #define BN_MP_CMP_MAG_C + #define BN_MP_EXCH_C + #define BN_S_MP_SUB_C + #define BN_MP_MUL_2D_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_GET_INT_C) + #endif + + #if defined(BN_MP_GET_LONG_C) + #endif + + #if defined(BN_MP_GET_LONG_LONG_C) + #endif + + #if defined(BN_MP_GROW_C) + #endif + + #if defined(BN_MP_IMPORT_C) + #define BN_MP_ZERO_C + #define BN_MP_MUL_2D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_INIT_C) + #endif + + #if defined(BN_MP_INIT_COPY_C) + #define BN_MP_INIT_SIZE_C + #define BN_MP_COPY_C + #endif + + #if defined(BN_MP_INIT_MULTI_C) + #define BN_MP_ERR_C + #define BN_MP_INIT_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_INIT_SET_C) + #define BN_MP_INIT_C + #define BN_MP_SET_C + #endif + + #if defined(BN_MP_INIT_SET_INT_C) + #define BN_MP_INIT_C + #define BN_MP_SET_INT_C + #endif + + #if defined(BN_MP_INIT_SIZE_C) + #define BN_MP_INIT_C + #endif + + #if defined(BN_MP_INVMOD_C) + #define BN_MP_ISZERO_C + #define BN_MP_ISODD_C + #define BN_FAST_MP_INVMOD_C + #define BN_MP_INVMOD_SLOW_C + #endif + + #if defined(BN_MP_INVMOD_SLOW_C) + #define BN_MP_ISZERO_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_MOD_C + #define BN_MP_COPY_C + #define BN_MP_ISEVEN_C + #define BN_MP_SET_C + #define BN_MP_DIV_2_C + #define BN_MP_ISODD_C + #define BN_MP_ADD_C + #define BN_MP_SUB_C + #define BN_MP_CMP_C + #define BN_MP_CMP_D_C + #define BN_MP_CMP_MAG_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_IS_SQUARE_C) + #define BN_MP_MOD_D_C + #define BN_MP_INIT_SET_INT_C + #define BN_MP_MOD_C + #define BN_MP_GET_INT_C + #define BN_MP_SQRT_C + #define BN_MP_SQR_C + #define BN_MP_CMP_MAG_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_JACOBI_C) + #define BN_MP_CMP_D_C + #define BN_MP_ISZERO_C + #define BN_MP_INIT_COPY_C + #define BN_MP_CNT_LSB_C + #define BN_MP_DIV_2D_C + #define BN_MP_MOD_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_KARATSUBA_MUL_C) + #define BN_MP_MUL_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_S_MP_ADD_C + #define BN_MP_ADD_C + #define BN_S_MP_SUB_C + #define BN_MP_LSHD_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_KARATSUBA_SQR_C) + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_SQR_C + #define BN_S_MP_ADD_C + #define BN_S_MP_SUB_C + #define BN_MP_LSHD_C + #define BN_MP_ADD_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_LCM_C) + #define BN_MP_INIT_MULTI_C + #define BN_MP_GCD_C + #define BN_MP_CMP_MAG_C + #define BN_MP_DIV_C + #define BN_MP_MUL_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_LSHD_C) + #define BN_MP_GROW_C + #define BN_MP_RSHD_C + #endif + + #if defined(BN_MP_MOD_C) + #define BN_MP_INIT_C + #define BN_MP_DIV_C + #define BN_MP_CLEAR_C + #define BN_MP_ISZERO_C + #define BN_MP_EXCH_C + #define BN_MP_ADD_C + #endif + + #if defined(BN_MP_MOD_2D_C) + #define BN_MP_ZERO_C + #define BN_MP_COPY_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_MOD_D_C) + #define BN_MP_DIV_D_C + #endif + + #if defined(BN_MP_MONTGOMERY_CALC_NORMALIZATION_C) + #define BN_MP_COUNT_BITS_C + #define BN_MP_2EXPT_C + #define BN_MP_SET_C + #define BN_MP_MUL_2_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_MONTGOMERY_REDUCE_C) + #define BN_FAST_MP_MONTGOMERY_REDUCE_C + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #define BN_MP_RSHD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_MONTGOMERY_SETUP_C) + #endif + + #if defined(BN_MP_MUL_C) + #define BN_MP_TOOM_MUL_C + #define BN_MP_KARATSUBA_MUL_C + #define BN_FAST_S_MP_MUL_DIGS_C + #define BN_S_MP_MUL_C + #define BN_S_MP_MUL_DIGS_C + #endif + + #if defined(BN_MP_MUL_2_C) + #define BN_MP_GROW_C + #endif + + #if defined(BN_MP_MUL_2D_C) + #define BN_MP_COPY_C + #define BN_MP_GROW_C + #define BN_MP_LSHD_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_MUL_D_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_MULMOD_C) + #define BN_MP_INIT_C + #define BN_MP_MUL_C + #define BN_MP_CLEAR_C + #define BN_MP_MOD_C + #endif + + #if defined(BN_MP_N_ROOT_C) + #define BN_MP_N_ROOT_EX_C + #endif + + #if defined(BN_MP_N_ROOT_EX_C) + #define BN_MP_INIT_C + #define BN_MP_SET_C + #define BN_MP_COPY_C + #define BN_MP_EXPT_D_EX_C + #define BN_MP_MUL_C + #define BN_MP_SUB_C + #define BN_MP_MUL_D_C + #define BN_MP_DIV_C + #define BN_MP_CMP_C + #define BN_MP_SUB_D_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_NEG_C) + #define BN_MP_COPY_C + #define BN_MP_ISZERO_C + #endif + + #if defined(BN_MP_OR_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_FERMAT_C) + #define BN_MP_CMP_D_C + #define BN_MP_INIT_C + #define BN_MP_EXPTMOD_C + #define BN_MP_CMP_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_IS_DIVISIBLE_C) + #define BN_MP_MOD_D_C + #endif + + #if defined(BN_MP_PRIME_IS_PRIME_C) + #define BN_MP_CMP_D_C + #define BN_MP_PRIME_IS_DIVISIBLE_C + #define BN_MP_INIT_C + #define BN_MP_SET_C + #define BN_MP_PRIME_MILLER_RABIN_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_MILLER_RABIN_C) + #define BN_MP_CMP_D_C + #define BN_MP_INIT_COPY_C + #define BN_MP_SUB_D_C + #define BN_MP_CNT_LSB_C + #define BN_MP_DIV_2D_C + #define BN_MP_EXPTMOD_C + #define BN_MP_CMP_C + #define BN_MP_SQRMOD_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_NEXT_PRIME_C) + #define BN_MP_CMP_D_C + #define BN_MP_SET_C + #define BN_MP_SUB_D_C + #define BN_MP_ISEVEN_C + #define BN_MP_MOD_D_C + #define BN_MP_INIT_C + #define BN_MP_ADD_D_C + #define BN_MP_PRIME_MILLER_RABIN_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_RABIN_MILLER_TRIALS_C) + #endif + + #if defined(BN_MP_PRIME_RANDOM_EX_C) + #define BN_MP_READ_UNSIGNED_BIN_C + #define BN_MP_PRIME_IS_PRIME_C + #define BN_MP_SUB_D_C + #define BN_MP_DIV_2_C + #define BN_MP_MUL_2_C + #define BN_MP_ADD_D_C + #endif + + #if defined(BN_MP_RADIX_SIZE_C) + #define BN_MP_ISZERO_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_INIT_COPY_C + #define BN_MP_DIV_D_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_RADIX_SMAP_C) + #define BN_MP_S_RMAP_C + #endif + + #if defined(BN_MP_RAND_C) + #define BN_MP_ZERO_C + #define BN_MP_ADD_D_C + #define BN_MP_LSHD_C + #endif + + #if defined(BN_MP_READ_RADIX_C) + #define BN_MP_ZERO_C + #define BN_MP_S_RMAP_C + #define BN_MP_MUL_D_C + #define BN_MP_ADD_D_C + #define BN_MP_ISZERO_C + #endif + + #if defined(BN_MP_READ_SIGNED_BIN_C) + #define BN_MP_READ_UNSIGNED_BIN_C + #endif + + #if defined(BN_MP_READ_UNSIGNED_BIN_C) + #define BN_MP_GROW_C + #define BN_MP_ZERO_C + #define BN_MP_MUL_2D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_REDUCE_C) + #define BN_MP_REDUCE_SETUP_C + #define BN_MP_INIT_COPY_C + #define BN_MP_RSHD_C + #define BN_MP_MUL_C + #define BN_S_MP_MUL_HIGH_DIGS_C + #define BN_FAST_S_MP_MUL_HIGH_DIGS_C + #define BN_MP_MOD_2D_C + #define BN_S_MP_MUL_DIGS_C + #define BN_MP_SUB_C + #define BN_MP_CMP_D_C + #define BN_MP_SET_C + #define BN_MP_LSHD_C + #define BN_MP_ADD_C + #define BN_MP_CMP_C + #define BN_S_MP_SUB_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_REDUCE_2K_C) + #define BN_MP_INIT_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_DIV_2D_C + #define BN_MP_MUL_D_C + #define BN_S_MP_ADD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_REDUCE_2K_L_C) + #define BN_MP_INIT_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_DIV_2D_C + #define BN_MP_MUL_C + #define BN_S_MP_ADD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_REDUCE_2K_SETUP_C) + #define BN_MP_INIT_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_2EXPT_C + #define BN_MP_CLEAR_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_REDUCE_2K_SETUP_L_C) + #define BN_MP_INIT_C + #define BN_MP_2EXPT_C + #define BN_MP_COUNT_BITS_C + #define BN_S_MP_SUB_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_REDUCE_IS_2K_C) + #define BN_MP_REDUCE_2K_C + #define BN_MP_COUNT_BITS_C + #endif + + #if defined(BN_MP_REDUCE_IS_2K_L_C) + #endif + + #if defined(BN_MP_REDUCE_SETUP_C) + #define BN_MP_2EXPT_C + #define BN_MP_DIV_C + #endif + + #if defined(BN_MP_RSHD_C) + #define BN_MP_ZERO_C + #endif + + #if defined(BN_MP_SET_C) + #define BN_MP_ZERO_C + #endif + + #if defined(BN_MP_SET_INT_C) + #define BN_MP_ZERO_C + #define BN_MP_MUL_2D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_SET_LONG_C) + #endif + + #if defined(BN_MP_SET_LONG_LONG_C) + #endif + + #if defined(BN_MP_SHRINK_C) + #endif + + #if defined(BN_MP_SIGNED_BIN_SIZE_C) + #define BN_MP_UNSIGNED_BIN_SIZE_C + #endif + + #if defined(BN_MP_SQR_C) + #define BN_MP_TOOM_SQR_C + #define BN_MP_KARATSUBA_SQR_C + #define BN_FAST_S_MP_SQR_C + #define BN_S_MP_SQR_C + #endif + + #if defined(BN_MP_SQRMOD_C) + #define BN_MP_INIT_C + #define BN_MP_SQR_C + #define BN_MP_CLEAR_C + #define BN_MP_MOD_C + #endif + + #if defined(BN_MP_SQRT_C) + #define BN_MP_N_ROOT_C + #define BN_MP_ISZERO_C + #define BN_MP_ZERO_C + #define BN_MP_INIT_COPY_C + #define BN_MP_RSHD_C + #define BN_MP_DIV_C + #define BN_MP_ADD_C + #define BN_MP_DIV_2_C + #define BN_MP_CMP_MAG_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_SQRTMOD_PRIME_C) + #define BN_MP_CMP_D_C + #define BN_MP_ZERO_C + #define BN_MP_JACOBI_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_MOD_D_C + #define BN_MP_ADD_D_C + #define BN_MP_DIV_2_C + #define BN_MP_EXPTMOD_C + #define BN_MP_COPY_C + #define BN_MP_SUB_D_C + #define BN_MP_ISEVEN_C + #define BN_MP_SET_INT_C + #define BN_MP_SQRMOD_C + #define BN_MP_MULMOD_C + #define BN_MP_SET_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_SUB_C) + #define BN_S_MP_ADD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_SUB_D_C) + #define BN_MP_GROW_C + #define BN_MP_ADD_D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_SUBMOD_C) + #define BN_MP_INIT_C + #define BN_MP_SUB_C + #define BN_MP_CLEAR_C + #define BN_MP_MOD_C + #endif + + #if defined(BN_MP_TO_SIGNED_BIN_C) + #define BN_MP_TO_UNSIGNED_BIN_C + #endif + + #if defined(BN_MP_TO_SIGNED_BIN_N_C) + #define BN_MP_SIGNED_BIN_SIZE_C + #define BN_MP_TO_SIGNED_BIN_C + #endif + + #if defined(BN_MP_TO_UNSIGNED_BIN_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_ISZERO_C + #define BN_MP_DIV_2D_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_TO_UNSIGNED_BIN_N_C) + #define BN_MP_UNSIGNED_BIN_SIZE_C + #define BN_MP_TO_UNSIGNED_BIN_C + #endif + + #if defined(BN_MP_TOOM_MUL_C) + #define BN_MP_INIT_MULTI_C + #define BN_MP_MOD_2D_C + #define BN_MP_COPY_C + #define BN_MP_RSHD_C + #define BN_MP_MUL_C + #define BN_MP_MUL_2_C + #define BN_MP_ADD_C + #define BN_MP_SUB_C + #define BN_MP_DIV_2_C + #define BN_MP_MUL_2D_C + #define BN_MP_MUL_D_C + #define BN_MP_DIV_3_C + #define BN_MP_LSHD_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_TOOM_SQR_C) + #define BN_MP_INIT_MULTI_C + #define BN_MP_MOD_2D_C + #define BN_MP_COPY_C + #define BN_MP_RSHD_C + #define BN_MP_SQR_C + #define BN_MP_MUL_2_C + #define BN_MP_ADD_C + #define BN_MP_SUB_C + #define BN_MP_DIV_2_C + #define BN_MP_MUL_2D_C + #define BN_MP_MUL_D_C + #define BN_MP_DIV_3_C + #define BN_MP_LSHD_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_TORADIX_C) + #define BN_MP_ISZERO_C + #define BN_MP_INIT_COPY_C + #define BN_MP_DIV_D_C + #define BN_MP_CLEAR_C + #define BN_MP_S_RMAP_C + #endif + + #if defined(BN_MP_TORADIX_N_C) + #define BN_MP_ISZERO_C + #define BN_MP_INIT_COPY_C + #define BN_MP_DIV_D_C + #define BN_MP_CLEAR_C + #define BN_MP_S_RMAP_C + #endif + + #if defined(BN_MP_UNSIGNED_BIN_SIZE_C) + #define BN_MP_COUNT_BITS_C + #endif + + #if defined(BN_MP_XOR_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_ZERO_C) + #endif + + #if defined(BN_PRIME_TAB_C) + #endif + + #if defined(BN_REVERSE_C) + #endif + + #if defined(BN_S_MP_ADD_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_S_MP_EXPTMOD_C) + #define BN_MP_COUNT_BITS_C + #define BN_MP_INIT_C + #define BN_MP_CLEAR_C + #define BN_MP_REDUCE_SETUP_C + #define BN_MP_REDUCE_C + #define BN_MP_REDUCE_2K_SETUP_L_C + #define BN_MP_REDUCE_2K_L_C + #define BN_MP_MOD_C + #define BN_MP_COPY_C + #define BN_MP_SQR_C + #define BN_MP_MUL_C + #define BN_MP_SET_C + #define BN_MP_EXCH_C + #endif + + #if defined(BN_S_MP_MUL_DIGS_C) + #define BN_FAST_S_MP_MUL_DIGS_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_S_MP_MUL_HIGH_DIGS_C) + #define BN_FAST_S_MP_MUL_HIGH_DIGS_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_S_MP_SQR_C) + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_S_MP_SUB_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BNCORE_C) + #endif + + #ifdef LTM3 + #define LTM_LAST + #endif +/* super class file for PK algos */ + +/* default ... include all MPI */ +#define LTM_ALL + +/* RSA only (does not support DH/DSA/ECC) */ +/* #define SC_RSA_1 */ + +/* For reference.... On an Athlon64 optimizing for speed... + + LTM's mpi.o with all functions [striped] is 142KiB in size. + + */ + +/* Works for RSA only, mpi.o is 68KiB */ +#ifdef SC_RSA_1 + #define BN_MP_SHRINK_C + #define BN_MP_LCM_C + #define BN_MP_PRIME_RANDOM_EX_C + #define BN_MP_INVMOD_C + #define BN_MP_GCD_C + #define BN_MP_MOD_C + #define BN_MP_MULMOD_C + #define BN_MP_ADDMOD_C + #define BN_MP_EXPTMOD_C + #define BN_MP_SET_INT_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_CLEAR_MULTI_C + #define BN_MP_UNSIGNED_BIN_SIZE_C + #define BN_MP_TO_UNSIGNED_BIN_C + #define BN_MP_MOD_D_C + #define BN_MP_PRIME_RABIN_MILLER_TRIALS_C + #define BN_REVERSE_C + #define BN_PRIME_TAB_C + +/* other modifiers */ + #define BN_MP_DIV_SMALL /* Slower division, not critical */ + +/* here we are on the last pass so we turn things off. The functions classes are still there + * but we remove them specifically from the build. This also invokes tweaks in functions + * like removing support for even moduli, etc... + */ + #ifdef LTM_LAST + #undef BN_MP_TOOM_MUL_C + #undef BN_MP_TOOM_SQR_C + #undef BN_MP_KARATSUBA_MUL_C + #undef BN_MP_KARATSUBA_SQR_C + #undef BN_MP_REDUCE_C + #undef BN_MP_REDUCE_SETUP_C + #undef BN_MP_DR_IS_MODULUS_C + #undef BN_MP_DR_SETUP_C + #undef BN_MP_DR_REDUCE_C + #undef BN_MP_REDUCE_IS_2K_C + #undef BN_MP_REDUCE_2K_SETUP_C + #undef BN_MP_REDUCE_2K_C + #undef BN_S_MP_EXPTMOD_C + #undef BN_MP_DIV_3_C + #undef BN_S_MP_MUL_HIGH_DIGS_C + #undef BN_FAST_S_MP_MUL_HIGH_DIGS_C + #undef BN_FAST_MP_INVMOD_C + +/* To safely undefine these you have to make sure your RSA key won't exceed the Comba threshold + * which is roughly 255 digits [7140 bits for 32-bit machines, 15300 bits for 64-bit machines] + * which means roughly speaking you can handle upto 2536-bit RSA keys with these defined without + * trouble. + */ + #undef BN_S_MP_MUL_DIGS_C + #undef BN_S_MP_SQR_C + #undef BN_MP_MONTGOMERY_REDUCE_C + #endif +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ +#if !(defined(LTM1) && defined(LTM2) && defined(LTM3)) + #if defined(LTM2) + #define LTM3 + #endif + #if defined(LTM1) + #define LTM2 + #endif + #define LTM1 + + #if defined(LTM_ALL) + #define BN_ERROR_C + #define BN_FAST_MP_INVMOD_C + #define BN_FAST_MP_MONTGOMERY_REDUCE_C + #define BN_FAST_S_MP_MUL_DIGS_C + #define BN_FAST_S_MP_MUL_HIGH_DIGS_C + #define BN_FAST_S_MP_SQR_C + #define BN_MP_2EXPT_C + #define BN_MP_ABS_C + #define BN_MP_ADD_C + #define BN_MP_ADD_D_C + #define BN_MP_ADDMOD_C + #define BN_MP_AND_C + #define BN_MP_CLAMP_C + #define BN_MP_CLEAR_C + #define BN_MP_CLEAR_MULTI_C + #define BN_MP_CMP_C + #define BN_MP_CMP_D_C + #define BN_MP_CMP_MAG_C + #define BN_MP_CNT_LSB_C + #define BN_MP_COPY_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_DIV_C + #define BN_MP_DIV_2_C + #define BN_MP_DIV_2D_C + #define BN_MP_DIV_3_C + #define BN_MP_DIV_D_C + #define BN_MP_DR_IS_MODULUS_C + #define BN_MP_DR_REDUCE_C + #define BN_MP_DR_SETUP_C + #define BN_MP_EXCH_C + #define BN_MP_EXPORT_C + #define BN_MP_EXPT_D_C + #define BN_MP_EXPT_D_EX_C + #define BN_MP_EXPTMOD_C + #define BN_MP_EXPTMOD_FAST_C + #define BN_MP_EXTEUCLID_C + #define BN_MP_FREAD_C + #define BN_MP_FWRITE_C + #define BN_MP_GCD_C + #define BN_MP_GET_INT_C + #define BN_MP_GET_LONG_C + #define BN_MP_GET_LONG_LONG_C + #define BN_MP_GROW_C + #define BN_MP_IMPORT_C + #define BN_MP_INIT_C + #define BN_MP_INIT_COPY_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_INIT_SET_C + #define BN_MP_INIT_SET_INT_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_INVMOD_C + #define BN_MP_INVMOD_SLOW_C + #define BN_MP_IS_SQUARE_C + #define BN_MP_JACOBI_C + #define BN_MP_KARATSUBA_MUL_C + #define BN_MP_KARATSUBA_SQR_C + #define BN_MP_LCM_C + #define BN_MP_LSHD_C + #define BN_MP_MOD_C + #define BN_MP_MOD_2D_C + #define BN_MP_MOD_D_C + #define BN_MP_MONTGOMERY_CALC_NORMALIZATION_C + #define BN_MP_MONTGOMERY_REDUCE_C + #define BN_MP_MONTGOMERY_SETUP_C + #define BN_MP_MUL_C + #define BN_MP_MUL_2_C + #define BN_MP_MUL_2D_C + #define BN_MP_MUL_D_C + #define BN_MP_MULMOD_C + #define BN_MP_N_ROOT_C + #define BN_MP_N_ROOT_EX_C + #define BN_MP_NEG_C + #define BN_MP_OR_C + #define BN_MP_PRIME_FERMAT_C + #define BN_MP_PRIME_IS_DIVISIBLE_C + #define BN_MP_PRIME_IS_PRIME_C + #define BN_MP_PRIME_MILLER_RABIN_C + #define BN_MP_PRIME_NEXT_PRIME_C + #define BN_MP_PRIME_RABIN_MILLER_TRIALS_C + #define BN_MP_PRIME_RANDOM_EX_C + #define BN_MP_RADIX_SIZE_C + #define BN_MP_RADIX_SMAP_C + #define BN_MP_RAND_C + #define BN_MP_READ_RADIX_C + #define BN_MP_READ_SIGNED_BIN_C + #define BN_MP_READ_UNSIGNED_BIN_C + #define BN_MP_REDUCE_C + #define BN_MP_REDUCE_2K_C + #define BN_MP_REDUCE_2K_L_C + #define BN_MP_REDUCE_2K_SETUP_C + #define BN_MP_REDUCE_2K_SETUP_L_C + #define BN_MP_REDUCE_IS_2K_C + #define BN_MP_REDUCE_IS_2K_L_C + #define BN_MP_REDUCE_SETUP_C + #define BN_MP_RSHD_C + #define BN_MP_SET_C + #define BN_MP_SET_INT_C + #define BN_MP_SET_LONG_C + #define BN_MP_SET_LONG_LONG_C + #define BN_MP_SHRINK_C + #define BN_MP_SIGNED_BIN_SIZE_C + #define BN_MP_SQR_C + #define BN_MP_SQRMOD_C + #define BN_MP_SQRT_C + #define BN_MP_SQRTMOD_PRIME_C + #define BN_MP_SUB_C + #define BN_MP_SUB_D_C + #define BN_MP_SUBMOD_C + #define BN_MP_TO_SIGNED_BIN_C + #define BN_MP_TO_SIGNED_BIN_N_C + #define BN_MP_TO_UNSIGNED_BIN_C + #define BN_MP_TO_UNSIGNED_BIN_N_C + #define BN_MP_TOOM_MUL_C + #define BN_MP_TOOM_SQR_C + #define BN_MP_TORADIX_C + #define BN_MP_TORADIX_N_C + #define BN_MP_UNSIGNED_BIN_SIZE_C + #define BN_MP_XOR_C + #define BN_MP_ZERO_C + #define BN_PRIME_TAB_C + #define BN_REVERSE_C + #define BN_S_MP_ADD_C + #define BN_S_MP_EXPTMOD_C + #define BN_S_MP_MUL_DIGS_C + #define BN_S_MP_MUL_HIGH_DIGS_C + #define BN_S_MP_SQR_C + #define BN_S_MP_SUB_C + #define BNCORE_C + #endif + + #if defined(BN_ERROR_C) + #define BN_MP_ERROR_TO_STRING_C + #endif + + #if defined(BN_FAST_MP_INVMOD_C) + #define BN_MP_ISEVEN_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_COPY_C + #define BN_MP_MOD_C + #define BN_MP_SET_C + #define BN_MP_DIV_2_C + #define BN_MP_ISODD_C + #define BN_MP_SUB_C + #define BN_MP_CMP_C + #define BN_MP_ISZERO_C + #define BN_MP_CMP_D_C + #define BN_MP_ADD_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_FAST_MP_MONTGOMERY_REDUCE_C) + #define BN_MP_GROW_C + #define BN_MP_RSHD_C + #define BN_MP_CLAMP_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_FAST_S_MP_MUL_DIGS_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_FAST_S_MP_MUL_HIGH_DIGS_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_FAST_S_MP_SQR_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_2EXPT_C) + #define BN_MP_ZERO_C + #define BN_MP_GROW_C + #endif + + #if defined(BN_MP_ABS_C) + #define BN_MP_COPY_C + #endif + + #if defined(BN_MP_ADD_C) + #define BN_S_MP_ADD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_ADD_D_C) + #define BN_MP_GROW_C + #define BN_MP_SUB_D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_ADDMOD_C) + #define BN_MP_INIT_C + #define BN_MP_ADD_C + #define BN_MP_CLEAR_C + #define BN_MP_MOD_C + #endif + + #if defined(BN_MP_AND_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_CLAMP_C) + #endif + + #if defined(BN_MP_CLEAR_C) + #endif + + #if defined(BN_MP_CLEAR_MULTI_C) + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_CMP_C) + #define BN_MP_CMP_MAG_C + #endif + + #if defined(BN_MP_CMP_D_C) + #endif + + #if defined(BN_MP_CMP_MAG_C) + #endif + + #if defined(BN_MP_CNT_LSB_C) + #define BN_MP_ISZERO_C + #endif + + #if defined(BN_MP_COPY_C) + #define BN_MP_GROW_C + #endif + + #if defined(BN_MP_COUNT_BITS_C) + #endif + + #if defined(BN_MP_DIV_C) + #define BN_MP_ISZERO_C + #define BN_MP_CMP_MAG_C + #define BN_MP_COPY_C + #define BN_MP_ZERO_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_SET_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_ABS_C + #define BN_MP_MUL_2D_C + #define BN_MP_CMP_C + #define BN_MP_SUB_C + #define BN_MP_ADD_C + #define BN_MP_DIV_2D_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_MULTI_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_INIT_C + #define BN_MP_INIT_COPY_C + #define BN_MP_LSHD_C + #define BN_MP_RSHD_C + #define BN_MP_MUL_D_C + #define BN_MP_CLAMP_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_DIV_2_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_DIV_2D_C) + #define BN_MP_COPY_C + #define BN_MP_ZERO_C + #define BN_MP_INIT_C + #define BN_MP_MOD_2D_C + #define BN_MP_CLEAR_C + #define BN_MP_RSHD_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #endif + + #if defined(BN_MP_DIV_3_C) + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_DIV_D_C) + #define BN_MP_ISZERO_C + #define BN_MP_COPY_C + #define BN_MP_DIV_2D_C + #define BN_MP_DIV_3_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_DR_IS_MODULUS_C) + #endif + + #if defined(BN_MP_DR_REDUCE_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_DR_SETUP_C) + #endif + + #if defined(BN_MP_EXCH_C) + #endif + + #if defined(BN_MP_EXPORT_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_DIV_2D_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_EXPT_D_C) + #define BN_MP_EXPT_D_EX_C + #endif + + #if defined(BN_MP_EXPT_D_EX_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_SET_C + #define BN_MP_MUL_C + #define BN_MP_CLEAR_C + #define BN_MP_SQR_C + #endif + + #if defined(BN_MP_EXPTMOD_C) + #define BN_MP_INIT_C + #define BN_MP_INVMOD_C + #define BN_MP_CLEAR_C + #define BN_MP_ABS_C + #define BN_MP_CLEAR_MULTI_C + #define BN_MP_REDUCE_IS_2K_L_C + #define BN_S_MP_EXPTMOD_C + #define BN_MP_DR_IS_MODULUS_C + #define BN_MP_REDUCE_IS_2K_C + #define BN_MP_ISODD_C + #define BN_MP_EXPTMOD_FAST_C + #endif + + #if defined(BN_MP_EXPTMOD_FAST_C) + #define BN_MP_COUNT_BITS_C + #define BN_MP_INIT_C + #define BN_MP_CLEAR_C + #define BN_MP_MONTGOMERY_SETUP_C + #define BN_FAST_MP_MONTGOMERY_REDUCE_C + #define BN_MP_MONTGOMERY_REDUCE_C + #define BN_MP_DR_SETUP_C + #define BN_MP_DR_REDUCE_C + #define BN_MP_REDUCE_2K_SETUP_C + #define BN_MP_REDUCE_2K_C + #define BN_MP_MONTGOMERY_CALC_NORMALIZATION_C + #define BN_MP_MULMOD_C + #define BN_MP_SET_C + #define BN_MP_MOD_C + #define BN_MP_COPY_C + #define BN_MP_SQR_C + #define BN_MP_MUL_C + #define BN_MP_EXCH_C + #endif + + #if defined(BN_MP_EXTEUCLID_C) + #define BN_MP_INIT_MULTI_C + #define BN_MP_SET_C + #define BN_MP_COPY_C + #define BN_MP_ISZERO_C + #define BN_MP_DIV_C + #define BN_MP_MUL_C + #define BN_MP_SUB_C + #define BN_MP_NEG_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_FREAD_C) + #define BN_MP_ZERO_C + #define BN_MP_S_RMAP_C + #define BN_MP_MUL_D_C + #define BN_MP_ADD_D_C + #define BN_MP_CMP_D_C + #endif + + #if defined(BN_MP_FWRITE_C) + #define BN_MP_RADIX_SIZE_C + #define BN_MP_TORADIX_C + #endif + + #if defined(BN_MP_GCD_C) + #define BN_MP_ISZERO_C + #define BN_MP_ABS_C + #define BN_MP_INIT_COPY_C + #define BN_MP_CNT_LSB_C + #define BN_MP_DIV_2D_C + #define BN_MP_CMP_MAG_C + #define BN_MP_EXCH_C + #define BN_S_MP_SUB_C + #define BN_MP_MUL_2D_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_GET_INT_C) + #endif + + #if defined(BN_MP_GET_LONG_C) + #endif + + #if defined(BN_MP_GET_LONG_LONG_C) + #endif + + #if defined(BN_MP_GROW_C) + #endif + + #if defined(BN_MP_IMPORT_C) + #define BN_MP_ZERO_C + #define BN_MP_MUL_2D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_INIT_C) + #endif + + #if defined(BN_MP_INIT_COPY_C) + #define BN_MP_INIT_SIZE_C + #define BN_MP_COPY_C + #endif + + #if defined(BN_MP_INIT_MULTI_C) + #define BN_MP_ERR_C + #define BN_MP_INIT_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_INIT_SET_C) + #define BN_MP_INIT_C + #define BN_MP_SET_C + #endif + + #if defined(BN_MP_INIT_SET_INT_C) + #define BN_MP_INIT_C + #define BN_MP_SET_INT_C + #endif + + #if defined(BN_MP_INIT_SIZE_C) + #define BN_MP_INIT_C + #endif + + #if defined(BN_MP_INVMOD_C) + #define BN_MP_ISZERO_C + #define BN_MP_ISODD_C + #define BN_FAST_MP_INVMOD_C + #define BN_MP_INVMOD_SLOW_C + #endif + + #if defined(BN_MP_INVMOD_SLOW_C) + #define BN_MP_ISZERO_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_MOD_C + #define BN_MP_COPY_C + #define BN_MP_ISEVEN_C + #define BN_MP_SET_C + #define BN_MP_DIV_2_C + #define BN_MP_ISODD_C + #define BN_MP_ADD_C + #define BN_MP_SUB_C + #define BN_MP_CMP_C + #define BN_MP_CMP_D_C + #define BN_MP_CMP_MAG_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_IS_SQUARE_C) + #define BN_MP_MOD_D_C + #define BN_MP_INIT_SET_INT_C + #define BN_MP_MOD_C + #define BN_MP_GET_INT_C + #define BN_MP_SQRT_C + #define BN_MP_SQR_C + #define BN_MP_CMP_MAG_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_JACOBI_C) + #define BN_MP_CMP_D_C + #define BN_MP_ISZERO_C + #define BN_MP_INIT_COPY_C + #define BN_MP_CNT_LSB_C + #define BN_MP_DIV_2D_C + #define BN_MP_MOD_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_KARATSUBA_MUL_C) + #define BN_MP_MUL_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_S_MP_ADD_C + #define BN_MP_ADD_C + #define BN_S_MP_SUB_C + #define BN_MP_LSHD_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_KARATSUBA_SQR_C) + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_SQR_C + #define BN_S_MP_ADD_C + #define BN_S_MP_SUB_C + #define BN_MP_LSHD_C + #define BN_MP_ADD_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_LCM_C) + #define BN_MP_INIT_MULTI_C + #define BN_MP_GCD_C + #define BN_MP_CMP_MAG_C + #define BN_MP_DIV_C + #define BN_MP_MUL_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_LSHD_C) + #define BN_MP_GROW_C + #define BN_MP_RSHD_C + #endif + + #if defined(BN_MP_MOD_C) + #define BN_MP_INIT_C + #define BN_MP_DIV_C + #define BN_MP_CLEAR_C + #define BN_MP_ISZERO_C + #define BN_MP_EXCH_C + #define BN_MP_ADD_C + #endif + + #if defined(BN_MP_MOD_2D_C) + #define BN_MP_ZERO_C + #define BN_MP_COPY_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_MOD_D_C) + #define BN_MP_DIV_D_C + #endif + + #if defined(BN_MP_MONTGOMERY_CALC_NORMALIZATION_C) + #define BN_MP_COUNT_BITS_C + #define BN_MP_2EXPT_C + #define BN_MP_SET_C + #define BN_MP_MUL_2_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_MONTGOMERY_REDUCE_C) + #define BN_FAST_MP_MONTGOMERY_REDUCE_C + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #define BN_MP_RSHD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_MONTGOMERY_SETUP_C) + #endif + + #if defined(BN_MP_MUL_C) + #define BN_MP_TOOM_MUL_C + #define BN_MP_KARATSUBA_MUL_C + #define BN_FAST_S_MP_MUL_DIGS_C + #define BN_S_MP_MUL_C + #define BN_S_MP_MUL_DIGS_C + #endif + + #if defined(BN_MP_MUL_2_C) + #define BN_MP_GROW_C + #endif + + #if defined(BN_MP_MUL_2D_C) + #define BN_MP_COPY_C + #define BN_MP_GROW_C + #define BN_MP_LSHD_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_MUL_D_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_MULMOD_C) + #define BN_MP_INIT_C + #define BN_MP_MUL_C + #define BN_MP_CLEAR_C + #define BN_MP_MOD_C + #endif + + #if defined(BN_MP_N_ROOT_C) + #define BN_MP_N_ROOT_EX_C + #endif + + #if defined(BN_MP_N_ROOT_EX_C) + #define BN_MP_INIT_C + #define BN_MP_SET_C + #define BN_MP_COPY_C + #define BN_MP_EXPT_D_EX_C + #define BN_MP_MUL_C + #define BN_MP_SUB_C + #define BN_MP_MUL_D_C + #define BN_MP_DIV_C + #define BN_MP_CMP_C + #define BN_MP_SUB_D_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_NEG_C) + #define BN_MP_COPY_C + #define BN_MP_ISZERO_C + #endif + + #if defined(BN_MP_OR_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_FERMAT_C) + #define BN_MP_CMP_D_C + #define BN_MP_INIT_C + #define BN_MP_EXPTMOD_C + #define BN_MP_CMP_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_IS_DIVISIBLE_C) + #define BN_MP_MOD_D_C + #endif + + #if defined(BN_MP_PRIME_IS_PRIME_C) + #define BN_MP_CMP_D_C + #define BN_MP_PRIME_IS_DIVISIBLE_C + #define BN_MP_INIT_C + #define BN_MP_SET_C + #define BN_MP_PRIME_MILLER_RABIN_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_MILLER_RABIN_C) + #define BN_MP_CMP_D_C + #define BN_MP_INIT_COPY_C + #define BN_MP_SUB_D_C + #define BN_MP_CNT_LSB_C + #define BN_MP_DIV_2D_C + #define BN_MP_EXPTMOD_C + #define BN_MP_CMP_C + #define BN_MP_SQRMOD_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_NEXT_PRIME_C) + #define BN_MP_CMP_D_C + #define BN_MP_SET_C + #define BN_MP_SUB_D_C + #define BN_MP_ISEVEN_C + #define BN_MP_MOD_D_C + #define BN_MP_INIT_C + #define BN_MP_ADD_D_C + #define BN_MP_PRIME_MILLER_RABIN_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_PRIME_RABIN_MILLER_TRIALS_C) + #endif + + #if defined(BN_MP_PRIME_RANDOM_EX_C) + #define BN_MP_READ_UNSIGNED_BIN_C + #define BN_MP_PRIME_IS_PRIME_C + #define BN_MP_SUB_D_C + #define BN_MP_DIV_2_C + #define BN_MP_MUL_2_C + #define BN_MP_ADD_D_C + #endif + + #if defined(BN_MP_RADIX_SIZE_C) + #define BN_MP_ISZERO_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_INIT_COPY_C + #define BN_MP_DIV_D_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_RADIX_SMAP_C) + #define BN_MP_S_RMAP_C + #endif + + #if defined(BN_MP_RAND_C) + #define BN_MP_ZERO_C + #define BN_MP_ADD_D_C + #define BN_MP_LSHD_C + #endif + + #if defined(BN_MP_READ_RADIX_C) + #define BN_MP_ZERO_C + #define BN_MP_S_RMAP_C + #define BN_MP_MUL_D_C + #define BN_MP_ADD_D_C + #define BN_MP_ISZERO_C + #endif + + #if defined(BN_MP_READ_SIGNED_BIN_C) + #define BN_MP_READ_UNSIGNED_BIN_C + #endif + + #if defined(BN_MP_READ_UNSIGNED_BIN_C) + #define BN_MP_GROW_C + #define BN_MP_ZERO_C + #define BN_MP_MUL_2D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_REDUCE_C) + #define BN_MP_REDUCE_SETUP_C + #define BN_MP_INIT_COPY_C + #define BN_MP_RSHD_C + #define BN_MP_MUL_C + #define BN_S_MP_MUL_HIGH_DIGS_C + #define BN_FAST_S_MP_MUL_HIGH_DIGS_C + #define BN_MP_MOD_2D_C + #define BN_S_MP_MUL_DIGS_C + #define BN_MP_SUB_C + #define BN_MP_CMP_D_C + #define BN_MP_SET_C + #define BN_MP_LSHD_C + #define BN_MP_ADD_C + #define BN_MP_CMP_C + #define BN_S_MP_SUB_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_REDUCE_2K_C) + #define BN_MP_INIT_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_DIV_2D_C + #define BN_MP_MUL_D_C + #define BN_S_MP_ADD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_REDUCE_2K_L_C) + #define BN_MP_INIT_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_DIV_2D_C + #define BN_MP_MUL_C + #define BN_S_MP_ADD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_REDUCE_2K_SETUP_C) + #define BN_MP_INIT_C + #define BN_MP_COUNT_BITS_C + #define BN_MP_2EXPT_C + #define BN_MP_CLEAR_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_REDUCE_2K_SETUP_L_C) + #define BN_MP_INIT_C + #define BN_MP_2EXPT_C + #define BN_MP_COUNT_BITS_C + #define BN_S_MP_SUB_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_REDUCE_IS_2K_C) + #define BN_MP_REDUCE_2K_C + #define BN_MP_COUNT_BITS_C + #endif + + #if defined(BN_MP_REDUCE_IS_2K_L_C) + #endif + + #if defined(BN_MP_REDUCE_SETUP_C) + #define BN_MP_2EXPT_C + #define BN_MP_DIV_C + #endif + + #if defined(BN_MP_RSHD_C) + #define BN_MP_ZERO_C + #endif + + #if defined(BN_MP_SET_C) + #define BN_MP_ZERO_C + #endif + + #if defined(BN_MP_SET_INT_C) + #define BN_MP_ZERO_C + #define BN_MP_MUL_2D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_SET_LONG_C) + #endif + + #if defined(BN_MP_SET_LONG_LONG_C) + #endif + + #if defined(BN_MP_SHRINK_C) + #endif + + #if defined(BN_MP_SIGNED_BIN_SIZE_C) + #define BN_MP_UNSIGNED_BIN_SIZE_C + #endif + + #if defined(BN_MP_SQR_C) + #define BN_MP_TOOM_SQR_C + #define BN_MP_KARATSUBA_SQR_C + #define BN_FAST_S_MP_SQR_C + #define BN_S_MP_SQR_C + #endif + + #if defined(BN_MP_SQRMOD_C) + #define BN_MP_INIT_C + #define BN_MP_SQR_C + #define BN_MP_CLEAR_C + #define BN_MP_MOD_C + #endif + + #if defined(BN_MP_SQRT_C) + #define BN_MP_N_ROOT_C + #define BN_MP_ISZERO_C + #define BN_MP_ZERO_C + #define BN_MP_INIT_COPY_C + #define BN_MP_RSHD_C + #define BN_MP_DIV_C + #define BN_MP_ADD_C + #define BN_MP_DIV_2_C + #define BN_MP_CMP_MAG_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_SQRTMOD_PRIME_C) + #define BN_MP_CMP_D_C + #define BN_MP_ZERO_C + #define BN_MP_JACOBI_C + #define BN_MP_INIT_MULTI_C + #define BN_MP_MOD_D_C + #define BN_MP_ADD_D_C + #define BN_MP_DIV_2_C + #define BN_MP_EXPTMOD_C + #define BN_MP_COPY_C + #define BN_MP_SUB_D_C + #define BN_MP_ISEVEN_C + #define BN_MP_SET_INT_C + #define BN_MP_SQRMOD_C + #define BN_MP_MULMOD_C + #define BN_MP_SET_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_SUB_C) + #define BN_S_MP_ADD_C + #define BN_MP_CMP_MAG_C + #define BN_S_MP_SUB_C + #endif + + #if defined(BN_MP_SUB_D_C) + #define BN_MP_GROW_C + #define BN_MP_ADD_D_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_MP_SUBMOD_C) + #define BN_MP_INIT_C + #define BN_MP_SUB_C + #define BN_MP_CLEAR_C + #define BN_MP_MOD_C + #endif + + #if defined(BN_MP_TO_SIGNED_BIN_C) + #define BN_MP_TO_UNSIGNED_BIN_C + #endif + + #if defined(BN_MP_TO_SIGNED_BIN_N_C) + #define BN_MP_SIGNED_BIN_SIZE_C + #define BN_MP_TO_SIGNED_BIN_C + #endif + + #if defined(BN_MP_TO_UNSIGNED_BIN_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_ISZERO_C + #define BN_MP_DIV_2D_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_TO_UNSIGNED_BIN_N_C) + #define BN_MP_UNSIGNED_BIN_SIZE_C + #define BN_MP_TO_UNSIGNED_BIN_C + #endif + + #if defined(BN_MP_TOOM_MUL_C) + #define BN_MP_INIT_MULTI_C + #define BN_MP_MOD_2D_C + #define BN_MP_COPY_C + #define BN_MP_RSHD_C + #define BN_MP_MUL_C + #define BN_MP_MUL_2_C + #define BN_MP_ADD_C + #define BN_MP_SUB_C + #define BN_MP_DIV_2_C + #define BN_MP_MUL_2D_C + #define BN_MP_MUL_D_C + #define BN_MP_DIV_3_C + #define BN_MP_LSHD_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_TOOM_SQR_C) + #define BN_MP_INIT_MULTI_C + #define BN_MP_MOD_2D_C + #define BN_MP_COPY_C + #define BN_MP_RSHD_C + #define BN_MP_SQR_C + #define BN_MP_MUL_2_C + #define BN_MP_ADD_C + #define BN_MP_SUB_C + #define BN_MP_DIV_2_C + #define BN_MP_MUL_2D_C + #define BN_MP_MUL_D_C + #define BN_MP_DIV_3_C + #define BN_MP_LSHD_C + #define BN_MP_CLEAR_MULTI_C + #endif + + #if defined(BN_MP_TORADIX_C) + #define BN_MP_ISZERO_C + #define BN_MP_INIT_COPY_C + #define BN_MP_DIV_D_C + #define BN_MP_CLEAR_C + #define BN_MP_S_RMAP_C + #endif + + #if defined(BN_MP_TORADIX_N_C) + #define BN_MP_ISZERO_C + #define BN_MP_INIT_COPY_C + #define BN_MP_DIV_D_C + #define BN_MP_CLEAR_C + #define BN_MP_S_RMAP_C + #endif + + #if defined(BN_MP_UNSIGNED_BIN_SIZE_C) + #define BN_MP_COUNT_BITS_C + #endif + + #if defined(BN_MP_XOR_C) + #define BN_MP_INIT_COPY_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_MP_ZERO_C) + #endif + + #if defined(BN_PRIME_TAB_C) + #endif + + #if defined(BN_REVERSE_C) + #endif + + #if defined(BN_S_MP_ADD_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BN_S_MP_EXPTMOD_C) + #define BN_MP_COUNT_BITS_C + #define BN_MP_INIT_C + #define BN_MP_CLEAR_C + #define BN_MP_REDUCE_SETUP_C + #define BN_MP_REDUCE_C + #define BN_MP_REDUCE_2K_SETUP_L_C + #define BN_MP_REDUCE_2K_L_C + #define BN_MP_MOD_C + #define BN_MP_COPY_C + #define BN_MP_SQR_C + #define BN_MP_MUL_C + #define BN_MP_SET_C + #define BN_MP_EXCH_C + #endif + + #if defined(BN_S_MP_MUL_DIGS_C) + #define BN_FAST_S_MP_MUL_DIGS_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_S_MP_MUL_HIGH_DIGS_C) + #define BN_FAST_S_MP_MUL_HIGH_DIGS_C + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_S_MP_SQR_C) + #define BN_MP_INIT_SIZE_C + #define BN_MP_CLAMP_C + #define BN_MP_EXCH_C + #define BN_MP_CLEAR_C + #endif + + #if defined(BN_S_MP_SUB_C) + #define BN_MP_GROW_C + #define BN_MP_CLAMP_C + #endif + + #if defined(BNCORE_C) + #endif + + #ifdef LTM3 + #define LTM_LAST + #endif +#else + #define LTM_LAST +#endif +#else + #define LTM_LAST +#endif + +#ifdef __cplusplus +extern "C" { +#endif + +/* detect 64-bit mode if possible */ +#if defined(__x86_64__) + #if !(defined(MP_32BIT) || defined(MP_16BIT) || defined(MP_8BIT)) + #define MP_64BIT + #endif +#endif + +/* some default configurations. + * + * A "mp_digit" must be able to hold DIGIT_BIT + 1 bits + * A "mp_word" must be able to hold 2*DIGIT_BIT + 1 bits + * + * At the very least a mp_digit must be able to hold 7 bits + * [any size beyond that is ok provided it doesn't overflow the data type] + */ +#ifdef MP_8BIT +typedef uint8_t mp_digit; +typedef uint16_t mp_word; + #define MP_SIZEOF_MP_DIGIT 1 + #ifdef DIGIT_BIT + #error You must not define DIGIT_BIT when using MP_8BIT + #endif +#elif defined(MP_16BIT) +typedef uint16_t mp_digit; +typedef uint32_t mp_word; + #define MP_SIZEOF_MP_DIGIT 2 + #ifdef DIGIT_BIT + #error You must not define DIGIT_BIT when using MP_16BIT + #endif +#elif defined(MP_64BIT) +/* for GCC only on supported platforms */ + #ifndef CRYPT +typedef unsigned long long ulong64; +typedef signed long long long64; + #endif + +typedef uint64_t mp_digit; + #if defined(_WIN32) +typedef unsigned __int128 mp_word; + #elif defined(__GNUC__) +typedef unsigned long mp_word __attribute__ ((mode(TI))); + #else + +/* it seems you have a problem + * but we assume you can somewhere define your own uint128_t */ +typedef uint128_t mp_word; + #endif + + #define DIGIT_BIT 60 +#else +/* this is the default case, 28-bit digits */ + +/* this is to make porting into LibTomCrypt easier :-) */ + #ifndef CRYPT +typedef unsigned long long ulong64; +typedef signed long long long64; + #endif + +typedef uint32_t mp_digit; +typedef uint64_t mp_word; + + #ifdef MP_31BIT +/* this is an extension that uses 31-bit digits */ + #define DIGIT_BIT 31 + #else +/* default case is 28-bit digits, defines MP_28BIT as a handy macro to test */ + #define DIGIT_BIT 28 + #define MP_28BIT + #endif +#endif + +/* otherwise the bits per digit is calculated automatically from the size of a mp_digit */ +#ifndef DIGIT_BIT + #define DIGIT_BIT (((CHAR_BIT * MP_SIZEOF_MP_DIGIT) - 1)) /* bits per digit */ +typedef uint_least32_t mp_min_u32; +#else +typedef mp_digit mp_min_u32; +#endif + +/* platforms that can use a better rand function */ +#if defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__NetBSD__) || defined(__DragonFly__) + #define MP_USE_ALT_RAND 1 +#endif + +/* use arc4random on platforms that support it */ +#ifdef MP_USE_ALT_RAND + #define MP_GEN_RANDOM() arc4random() +#else + #define MP_GEN_RANDOM() rand() +#endif + +#define MP_DIGIT_BIT DIGIT_BIT +#define MP_MASK ((((mp_digit)1) << ((mp_digit)DIGIT_BIT)) - ((mp_digit)1)) +#define MP_DIGIT_MAX MP_MASK + +/* equalities */ +#define MP_LT -1 /* less than */ +#define MP_EQ 0 /* equal to */ +#define MP_GT 1 /* greater than */ + +#define MP_ZPOS 0 /* positive integer */ +#define MP_NEG 1 /* negative */ + +#define MP_OKAY 0 /* ok result */ +#define MP_MEM -2 /* out of mem */ +#define MP_VAL -3 /* invalid input */ +#define MP_RANGE MP_VAL + +#define MP_YES 1 /* yes response */ +#define MP_NO 0 /* no response */ + +/* Primality generation flags */ +#define LTM_PRIME_BBS 0x0001 /* BBS style prime */ +#define LTM_PRIME_SAFE 0x0002 /* Safe prime (p-1)/2 == prime */ +#define LTM_PRIME_2MSB_ON 0x0008 /* force 2nd MSB to 1 */ + +typedef int mp_err; + +/* you'll have to tune these... */ +extern int KARATSUBA_MUL_CUTOFF, + KARATSUBA_SQR_CUTOFF, + TOOM_MUL_CUTOFF, + TOOM_SQR_CUTOFF; + +/* define this to use lower memory usage routines (exptmods mostly) */ +/* #define MP_LOW_MEM */ + +/* default precision */ +#ifndef MP_PREC + #ifndef MP_LOW_MEM + #define MP_PREC 32 /* default digits of precision */ + #else + #define MP_PREC 8 /* default digits of precision */ + #endif +#endif + +/* size of comba arrays, should be at least 2 * 2**(BITS_PER_WORD - BITS_PER_DIGIT*2) */ +#define MP_WARRAY (1 << (((sizeof(mp_word) * CHAR_BIT) - (2 * DIGIT_BIT)) + 1)) + +/* the infamous mp_int structure */ +typedef struct { + int used, alloc, sign; + mp_digit *dp; +} mp_int; + +/* callback for mp_prime_random, should fill dst with random bytes and return how many read [upto len] */ +typedef int ltm_prime_callback (unsigned char *dst, int len, void *dat); + + +#define USED(m) ((m)->used) +#define DIGIT(m, k) ((m)->dp[(k)]) +#define SIGN(m) ((m)->sign) + +/* error code to char* string */ +const char *mp_error_to_string(int code); + +/* ---> init and deinit bignum functions <--- */ +/* init a bignum */ +int mp_init(mp_int *a); + +/* free a bignum */ +void mp_clear(mp_int *a); + +/* init a null terminated series of arguments */ +int mp_init_multi(mp_int *mp, ...); + +/* clear a null terminated series of arguments */ +void mp_clear_multi(mp_int *mp, ...); + +/* exchange two ints */ +void mp_exch(mp_int *a, mp_int *b); + +/* shrink ram required for a bignum */ +int mp_shrink(mp_int *a); + +/* grow an int to a given size */ +int mp_grow(mp_int *a, int size); + +/* init to a given number of digits */ +int mp_init_size(mp_int *a, int size); + +/* ---> Basic Manipulations <--- */ +#define mp_iszero(a) (((a)->used == 0) ? MP_YES : MP_NO) +#define mp_iseven(a) ((((a)->used > 0) && (((a)->dp[0] & 1u) == 0u)) ? MP_YES : MP_NO) +#define mp_isodd(a) ((((a)->used > 0) && (((a)->dp[0] & 1u) == 1u)) ? MP_YES : MP_NO) +#define mp_isneg(a) (((a)->sign != MP_ZPOS) ? MP_YES : MP_NO) + +/* set to zero */ +void mp_zero(mp_int *a); + +/* set to a digit */ +void mp_set(mp_int *a, mp_digit b); + +/* set a 32-bit const */ +int mp_set_int(mp_int *a, unsigned long b); + +/* set a platform dependent unsigned long value */ +int mp_set_long(mp_int *a, unsigned long b); + +/* set a platform dependent unsigned long long value */ +int mp_set_long_long(mp_int *a, unsigned long long b); + +/* get a 32-bit value */ +unsigned long mp_get_int(mp_int *a); + +/* get a platform dependent unsigned long value */ +unsigned long mp_get_long(mp_int *a); + +/* get a platform dependent unsigned long long value */ +unsigned long long mp_get_long_long(mp_int *a); + +/* initialize and set a digit */ +int mp_init_set(mp_int *a, mp_digit b); + +/* initialize and set 32-bit value */ +int mp_init_set_int(mp_int *a, unsigned long b); + +/* copy, b = a */ +int mp_copy(mp_int *a, mp_int *b); + +/* inits and copies, a = b */ +int mp_init_copy(mp_int *a, mp_int *b); + +/* trim unused digits */ +void mp_clamp(mp_int *a); + +/* import binary data */ +int mp_import(mp_int *rop, size_t count, int order, size_t size, int endian, size_t nails, const void *op); + +/* export binary data */ +int mp_export(void *rop, size_t *countp, int order, size_t size, int endian, size_t nails, mp_int *op); + +/* ---> digit manipulation <--- */ + +/* right shift by "b" digits */ +void mp_rshd(mp_int *a, int b); + +/* left shift by "b" digits */ +int mp_lshd(mp_int *a, int b); + +/* c = a / 2**b, implemented as c = a >> b */ +int mp_div_2d(mp_int *a, int b, mp_int *c, mp_int *d); + +/* b = a/2 */ +int mp_div_2(mp_int *a, mp_int *b); + +/* c = a * 2**b, implemented as c = a << b */ +int mp_mul_2d(mp_int *a, int b, mp_int *c); + +/* b = a*2 */ +int mp_mul_2(mp_int *a, mp_int *b); + +/* c = a mod 2**b */ +int mp_mod_2d(mp_int *a, int b, mp_int *c); + +/* computes a = 2**b */ +int mp_2expt(mp_int *a, int b); + +/* Counts the number of lsbs which are zero before the first zero bit */ +int mp_cnt_lsb(mp_int *a); + +/* I Love Earth! */ + +/* makes a pseudo-random int of a given size */ +int mp_rand(mp_int *a, int digits); + +/* ---> binary operations <--- */ +/* c = a XOR b */ +int mp_xor(mp_int *a, mp_int *b, mp_int *c); + +/* c = a OR b */ +int mp_or(mp_int *a, mp_int *b, mp_int *c); + +/* c = a AND b */ +int mp_and(mp_int *a, mp_int *b, mp_int *c); + +/* ---> Basic arithmetic <--- */ + +/* b = -a */ +int mp_neg(mp_int *a, mp_int *b); + +/* b = |a| */ +int mp_abs(mp_int *a, mp_int *b); + +/* compare a to b */ +int mp_cmp(mp_int *a, mp_int *b); + +/* compare |a| to |b| */ +int mp_cmp_mag(mp_int *a, mp_int *b); + +/* c = a + b */ +int mp_add(mp_int *a, mp_int *b, mp_int *c); + +/* c = a - b */ +int mp_sub(mp_int *a, mp_int *b, mp_int *c); + +/* c = a * b */ +int mp_mul(mp_int *a, mp_int *b, mp_int *c); + +/* b = a*a */ +int mp_sqr(mp_int *a, mp_int *b); + +/* a/b => cb + d == a */ +int mp_div(mp_int *a, mp_int *b, mp_int *c, mp_int *d); + +/* c = a mod b, 0 <= c < b */ +int mp_mod(mp_int *a, mp_int *b, mp_int *c); + +/* ---> single digit functions <--- */ + +/* compare against a single digit */ +int mp_cmp_d(mp_int *a, mp_digit b); + +/* c = a + b */ +int mp_add_d(mp_int *a, mp_digit b, mp_int *c); + +/* c = a - b */ +int mp_sub_d(mp_int *a, mp_digit b, mp_int *c); + +/* c = a * b */ +int mp_mul_d(mp_int *a, mp_digit b, mp_int *c); + +/* a/b => cb + d == a */ +int mp_div_d(mp_int *a, mp_digit b, mp_int *c, mp_digit *d); + +/* a/3 => 3c + d == a */ +int mp_div_3(mp_int *a, mp_int *c, mp_digit *d); + +/* c = a**b */ +int mp_expt_d(mp_int *a, mp_digit b, mp_int *c); +int mp_expt_d_ex(mp_int *a, mp_digit b, mp_int *c, int fast); + +/* c = a mod b, 0 <= c < b */ +int mp_mod_d(mp_int *a, mp_digit b, mp_digit *c); + +/* ---> number theory <--- */ + +/* d = a + b (mod c) */ +int mp_addmod(mp_int *a, mp_int *b, mp_int *c, mp_int *d); + +/* d = a - b (mod c) */ +int mp_submod(mp_int *a, mp_int *b, mp_int *c, mp_int *d); + +/* d = a * b (mod c) */ +int mp_mulmod(mp_int *a, mp_int *b, mp_int *c, mp_int *d); + +/* c = a * a (mod b) */ +int mp_sqrmod(mp_int *a, mp_int *b, mp_int *c); + +/* c = 1/a (mod b) */ +int mp_invmod(mp_int *a, mp_int *b, mp_int *c); + +/* c = (a, b) */ +int mp_gcd(mp_int *a, mp_int *b, mp_int *c); + +/* produces value such that U1*a + U2*b = U3 */ +int mp_exteuclid(mp_int *a, mp_int *b, mp_int *U1, mp_int *U2, mp_int *U3); + +/* c = [a, b] or (a*b)/(a, b) */ +int mp_lcm(mp_int *a, mp_int *b, mp_int *c); + +/* finds one of the b'th root of a, such that |c|**b <= |a| + * + * returns error if a < 0 and b is even + */ +int mp_n_root(mp_int *a, mp_digit b, mp_int *c); +int mp_n_root_ex(mp_int *a, mp_digit b, mp_int *c, int fast); + +/* special sqrt algo */ +int mp_sqrt(mp_int *arg, mp_int *ret); + +/* special sqrt (mod prime) */ +int mp_sqrtmod_prime(mp_int *arg, mp_int *prime, mp_int *ret); + +/* is number a square? */ +int mp_is_square(mp_int *arg, int *ret); + +/* computes the jacobi c = (a | n) (or Legendre if b is prime) */ +int mp_jacobi(mp_int *a, mp_int *n, int *c); + +/* used to setup the Barrett reduction for a given modulus b */ +int mp_reduce_setup(mp_int *a, mp_int *b); + +/* Barrett Reduction, computes a (mod b) with a precomputed value c + * + * Assumes that 0 < a <= b*b, note if 0 > a > -(b*b) then you can merely + * compute the reduction as -1 * mp_reduce(mp_abs(a)) [pseudo code]. + */ +int mp_reduce(mp_int *a, mp_int *b, mp_int *c); + +/* setups the montgomery reduction */ +int mp_montgomery_setup(mp_int *a, mp_digit *mp); + +/* computes a = B**n mod b without division or multiplication useful for + * normalizing numbers in a Montgomery system. + */ +int mp_montgomery_calc_normalization(mp_int *a, mp_int *b); + +/* computes x/R == x (mod N) via Montgomery Reduction */ +int mp_montgomery_reduce(mp_int *a, mp_int *m, mp_digit mp); + +/* returns 1 if a is a valid DR modulus */ +int mp_dr_is_modulus(mp_int *a); + +/* sets the value of "d" required for mp_dr_reduce */ +void mp_dr_setup(mp_int *a, mp_digit *d); + +/* reduces a modulo b using the Diminished Radix method */ +int mp_dr_reduce(mp_int *a, mp_int *b, mp_digit mp); + +/* returns true if a can be reduced with mp_reduce_2k */ +int mp_reduce_is_2k(mp_int *a); + +/* determines k value for 2k reduction */ +int mp_reduce_2k_setup(mp_int *a, mp_digit *d); + +/* reduces a modulo b where b is of the form 2**p - k [0 <= a] */ +int mp_reduce_2k(mp_int *a, mp_int *n, mp_digit d); + +/* returns true if a can be reduced with mp_reduce_2k_l */ +int mp_reduce_is_2k_l(mp_int *a); + +/* determines k value for 2k reduction */ +int mp_reduce_2k_setup_l(mp_int *a, mp_int *d); + +/* reduces a modulo b where b is of the form 2**p - k [0 <= a] */ +int mp_reduce_2k_l(mp_int *a, mp_int *n, mp_int *d); + +/* d = a**b (mod c) */ +int mp_exptmod(mp_int *a, mp_int *b, mp_int *c, mp_int *d); + +/* ---> Primes <--- */ + +/* number of primes */ +#ifdef MP_8BIT + #define PRIME_SIZE 31 +#else + #define PRIME_SIZE 256 +#endif + +/* table of first PRIME_SIZE primes */ +extern const mp_digit ltm_prime_tab[PRIME_SIZE]; + +/* result=1 if a is divisible by one of the first PRIME_SIZE primes */ +int mp_prime_is_divisible(mp_int *a, int *result); + +/* performs one Fermat test of "a" using base "b". + * Sets result to 0 if composite or 1 if probable prime + */ +int mp_prime_fermat(mp_int *a, mp_int *b, int *result); + +/* performs one Miller-Rabin test of "a" using base "b". + * Sets result to 0 if composite or 1 if probable prime + */ +int mp_prime_miller_rabin(mp_int *a, mp_int *b, int *result); + +/* This gives [for a given bit size] the number of trials required + * such that Miller-Rabin gives a prob of failure lower than 2^-96 + */ +int mp_prime_rabin_miller_trials(int size); + +/* performs t rounds of Miller-Rabin on "a" using the first + * t prime bases. Also performs an initial sieve of trial + * division. Determines if "a" is prime with probability + * of error no more than (1/4)**t. + * + * Sets result to 1 if probably prime, 0 otherwise + */ +int mp_prime_is_prime(mp_int *a, int t, int *result); + +/* finds the next prime after the number "a" using "t" trials + * of Miller-Rabin. + * + * bbs_style = 1 means the prime must be congruent to 3 mod 4 + */ +int mp_prime_next_prime(mp_int *a, int t, int bbs_style); + +/* makes a truly random prime of a given size (bytes), + * call with bbs = 1 if you want it to be congruent to 3 mod 4 + * + * You have to supply a callback which fills in a buffer with random bytes. "dat" is a parameter you can + * have passed to the callback (e.g. a state or something). This function doesn't use "dat" itself + * so it can be NULL + * + * The prime generated will be larger than 2^(8*size). + */ +#define mp_prime_random(a, t, size, bbs, cb, dat) mp_prime_random_ex(a, t, ((size) * 8) + 1, (bbs == 1) ? LTM_PRIME_BBS : 0, cb, dat) + +/* makes a truly random prime of a given size (bits), + * + * Flags are as follows: + * + * LTM_PRIME_BBS - make prime congruent to 3 mod 4 + * LTM_PRIME_SAFE - make sure (p-1)/2 is prime as well (implies LTM_PRIME_BBS) + * LTM_PRIME_2MSB_ON - make the 2nd highest bit one + * + * You have to supply a callback which fills in a buffer with random bytes. "dat" is a parameter you can + * have passed to the callback (e.g. a state or something). This function doesn't use "dat" itself + * so it can be NULL + * + */ +int mp_prime_random_ex(mp_int *a, int t, int size, int flags, ltm_prime_callback cb, void *dat); + +/* ---> radix conversion <--- */ +int mp_count_bits(mp_int *a); + +int mp_unsigned_bin_size(mp_int *a); +int mp_read_unsigned_bin(mp_int *a, const unsigned char *b, int c); +int mp_to_unsigned_bin(mp_int *a, unsigned char *b); +int mp_to_unsigned_bin_n(mp_int *a, unsigned char *b, unsigned long *outlen); + +int mp_signed_bin_size(mp_int *a); +int mp_read_signed_bin(mp_int *a, const unsigned char *b, int c); +int mp_to_signed_bin(mp_int *a, unsigned char *b); +int mp_to_signed_bin_n(mp_int *a, unsigned char *b, unsigned long *outlen); + +int mp_read_radix(mp_int *a, const char *str, int radix); +int mp_toradix(mp_int *a, char *str, int radix); +int mp_toradix_n(mp_int *a, char *str, int radix, int maxlen); +int mp_radix_size(mp_int *a, int radix, int *size); + +#ifndef LTM_NO_FILE +int mp_fread(mp_int *a, int radix, FILE *stream); +int mp_fwrite(mp_int *a, int radix, FILE *stream); +#endif + +#define mp_read_raw(mp, str, len) mp_read_signed_bin((mp), (str), (len)) +#define mp_raw_size(mp) mp_signed_bin_size(mp) +#define mp_toraw(mp, str) mp_to_signed_bin((mp), (str)) +#define mp_read_mag(mp, str, len) mp_read_unsigned_bin((mp), (str), (len)) +#define mp_mag_size(mp) mp_unsigned_bin_size(mp) +#define mp_tomag(mp, str) mp_to_unsigned_bin((mp), (str)) + +#define mp_tobinary(M, S) mp_toradix((M), (S), 2) +#define mp_tooctal(M, S) mp_toradix((M), (S), 8) +#define mp_todecimal(M, S) mp_toradix((M), (S), 10) +#define mp_tohex(M, S) mp_toradix((M), (S), 16) + +#ifdef __cplusplus +} +#endif +#endif + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://math.libtomcrypt.com + */ +#ifndef TOMMATH_PRIV_H_ +#define TOMMATH_PRIV_H_ + +#include + +#define MIN(x, y) (((x) < (y)) ? (x) : (y)) + +#define MAX(x, y) (((x) > (y)) ? (x) : (y)) + +#ifdef __cplusplus +extern "C" { +/* C++ compilers don't like assigning void * to mp_digit * */ + #define OPT_CAST(x) (x *) + +#else + +/* C on the other hand doesn't care */ + #define OPT_CAST(x) +#endif + +/* define heap macros */ +#ifndef XMALLOC +/* default to libc stuff */ + #define XMALLOC malloc + #define XFREE free + #define XREALLOC realloc + #define XCALLOC calloc +#else +/* prototypes for our heap functions */ +extern void *XMALLOC(size_t n); +extern void *XREALLOC(void *p, size_t n); +extern void *XCALLOC(size_t n, size_t s); +extern void XFREE(void *p); +#endif + +/* lowlevel functions, do not call! */ +int s_mp_add(mp_int *a, mp_int *b, mp_int *c); +int s_mp_sub(mp_int *a, mp_int *b, mp_int *c); + +#define s_mp_mul(a, b, c) s_mp_mul_digs(a, b, c, (a)->used + (b)->used + 1) +int fast_s_mp_mul_digs(mp_int *a, mp_int *b, mp_int *c, int digs); +int s_mp_mul_digs(mp_int *a, mp_int *b, mp_int *c, int digs); +int fast_s_mp_mul_high_digs(mp_int *a, mp_int *b, mp_int *c, int digs); +int s_mp_mul_high_digs(mp_int *a, mp_int *b, mp_int *c, int digs); +int fast_s_mp_sqr(mp_int *a, mp_int *b); +int s_mp_sqr(mp_int *a, mp_int *b); +int mp_karatsuba_mul(mp_int *a, mp_int *b, mp_int *c); +int mp_toom_mul(mp_int *a, mp_int *b, mp_int *c); +int mp_karatsuba_sqr(mp_int *a, mp_int *b); +int mp_toom_sqr(mp_int *a, mp_int *b); +int fast_mp_invmod(mp_int *a, mp_int *b, mp_int *c); +int mp_invmod_slow(mp_int *a, mp_int *b, mp_int *c); +int fast_mp_montgomery_reduce(mp_int *x, mp_int *n, mp_digit rho); +int mp_exptmod_fast(mp_int *G, mp_int *X, mp_int *P, mp_int *Y, int redmode); +int s_mp_exptmod(mp_int *G, mp_int *X, mp_int *P, mp_int *Y, int redmode); +void bn_reverse(unsigned char *s, int len); + +extern const char *mp_s_rmap; + +/* Fancy macro to set an MPI from another type. + * There are several things assumed: + * x is the counter and unsigned + * a is the pointer to the MPI + * b is the original value that should be set in the MPI. + */ +#define MP_SET_XLONG(func_name, type) \ + int func_name(mp_int * a, type b) \ + { \ + unsigned int x; \ + int res; \ + \ + mp_zero(a); \ + \ + /* set four bits at a time */ \ + for (x = 0; x < (sizeof(type) * 2u); x++) { \ + /* shift the number up four bits */ \ + if ((res = mp_mul_2d(a, 4, a)) != MP_OKAY) { \ + return res; \ + } \ + \ + /* OR in the top four bits of the source */ \ + a->dp[0] |= (b >> ((sizeof(type) * 8u) - 4u)) & 15u; \ + \ + /* shift the source up to the next four bits */ \ + b <<= 4; \ + \ + /* ensure that digits are not clamped off */ \ + a->used += 1; \ + } \ + mp_clamp(a); \ + return MP_OKAY; \ + } + +#ifdef __cplusplus +} +#endif +#endif + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + +#define BN_FAST_MP_INVMOD_C +#ifdef BN_FAST_MP_INVMOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* computes the modular inverse via binary extended euclidean algorithm, + * that is c = 1/a mod b + * + * Based on slow invmod except this is optimized for the case where b is + * odd as per HAC Note 14.64 on pp. 610 + */ +int fast_mp_invmod(mp_int *a, mp_int *b, mp_int *c) { + mp_int x, y, u, v, B, D; + int res, neg; + + /* 2. [modified] b must be odd */ + if (mp_iseven(b) == MP_YES) { + return MP_VAL; + } + + /* init all our temps */ + if ((res = mp_init_multi(&x, &y, &u, &v, &B, &D, NULL)) != MP_OKAY) { + return res; + } + + /* x == modulus, y == value to invert */ + if ((res = mp_copy(b, &x)) != MP_OKAY) { + goto LBL_ERR; + } + + /* we need y = |a| */ + if ((res = mp_mod(a, b, &y)) != MP_OKAY) { + goto LBL_ERR; + } + + /* 3. u=x, v=y, A=1, B=0, C=0,D=1 */ + if ((res = mp_copy(&x, &u)) != MP_OKAY) { + goto LBL_ERR; + } + if ((res = mp_copy(&y, &v)) != MP_OKAY) { + goto LBL_ERR; + } + mp_set(&D, 1); + +top: + /* 4. while u is even do */ + while (mp_iseven(&u) == MP_YES) { + /* 4.1 u = u/2 */ + if ((res = mp_div_2(&u, &u)) != MP_OKAY) { + goto LBL_ERR; + } + /* 4.2 if B is odd then */ + if (mp_isodd(&B) == MP_YES) { + if ((res = mp_sub(&B, &x, &B)) != MP_OKAY) { + goto LBL_ERR; + } + } + /* B = B/2 */ + if ((res = mp_div_2(&B, &B)) != MP_OKAY) { + goto LBL_ERR; + } + } + + /* 5. while v is even do */ + while (mp_iseven(&v) == MP_YES) { + /* 5.1 v = v/2 */ + if ((res = mp_div_2(&v, &v)) != MP_OKAY) { + goto LBL_ERR; + } + /* 5.2 if D is odd then */ + if (mp_isodd(&D) == MP_YES) { + /* D = (D-x)/2 */ + if ((res = mp_sub(&D, &x, &D)) != MP_OKAY) { + goto LBL_ERR; + } + } + /* D = D/2 */ + if ((res = mp_div_2(&D, &D)) != MP_OKAY) { + goto LBL_ERR; + } + } + + /* 6. if u >= v then */ + if (mp_cmp(&u, &v) != MP_LT) { + /* u = u - v, B = B - D */ + if ((res = mp_sub(&u, &v, &u)) != MP_OKAY) { + goto LBL_ERR; + } + + if ((res = mp_sub(&B, &D, &B)) != MP_OKAY) { + goto LBL_ERR; + } + } else { + /* v - v - u, D = D - B */ + if ((res = mp_sub(&v, &u, &v)) != MP_OKAY) { + goto LBL_ERR; + } + + if ((res = mp_sub(&D, &B, &D)) != MP_OKAY) { + goto LBL_ERR; + } + } + + /* if not zero goto step 4 */ + if (mp_iszero(&u) == MP_NO) { + goto top; + } + + /* now a = C, b = D, gcd == g*v */ + + /* if v != 1 then there is no inverse */ + if (mp_cmp_d(&v, 1) != MP_EQ) { + res = MP_VAL; + goto LBL_ERR; + } + + /* b is now the inverse */ + neg = a->sign; + while (D.sign == MP_NEG) { + if ((res = mp_add(&D, b, &D)) != MP_OKAY) { + goto LBL_ERR; + } + } + mp_exch(&D, c); + c->sign = neg; + res = MP_OKAY; + +LBL_ERR: mp_clear_multi(&x, &y, &u, &v, &B, &D, NULL); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_FAST_MP_MONTGOMERY_REDUCE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* computes xR**-1 == x (mod N) via Montgomery Reduction + * + * This is an optimized implementation of montgomery_reduce + * which uses the comba method to quickly calculate the columns of the + * reduction. + * + * Based on Algorithm 14.32 on pp.601 of HAC. + */ +int fast_mp_montgomery_reduce(mp_int *x, mp_int *n, mp_digit rho) { + int ix, res, olduse; + mp_word W[MP_WARRAY]; + + /* get old used count */ + olduse = x->used; + + /* grow a as required */ + if (x->alloc < (n->used + 1)) { + if ((res = mp_grow(x, n->used + 1)) != MP_OKAY) { + return res; + } + } + + /* first we have to get the digits of the input into + * an array of double precision words W[...] + */ + { + mp_word *_W; + mp_digit *tmpx; + + /* alias for the W[] array */ + _W = W; + + /* alias for the digits of x*/ + tmpx = x->dp; + + /* copy the digits of a into W[0..a->used-1] */ + for (ix = 0; ix < x->used; ix++) { + *_W++ = *tmpx++; + } + + /* zero the high words of W[a->used..m->used*2] */ + for ( ; ix < ((n->used * 2) + 1); ix++) { + *_W++ = 0; + } + } + + /* now we proceed to zero successive digits + * from the least significant upwards + */ + for (ix = 0; ix < n->used; ix++) { + /* mu = ai * m' mod b + * + * We avoid a double precision multiplication (which isn't required) + * by casting the value down to a mp_digit. Note this requires + * that W[ix-1] have the carry cleared (see after the inner loop) + */ + mp_digit mu; + mu = (mp_digit)(((W[ix] & MP_MASK) * rho) & MP_MASK); + + /* a = a + mu * m * b**i + * + * This is computed in place and on the fly. The multiplication + * by b**i is handled by offseting which columns the results + * are added to. + * + * Note the comba method normally doesn't handle carries in the + * inner loop In this case we fix the carry from the previous + * column since the Montgomery reduction requires digits of the + * result (so far) [see above] to work. This is + * handled by fixing up one carry after the inner loop. The + * carry fixups are done in order so after these loops the + * first m->used words of W[] have the carries fixed + */ + { + int iy; + mp_digit *tmpn; + mp_word *_W; + + /* alias for the digits of the modulus */ + tmpn = n->dp; + + /* Alias for the columns set by an offset of ix */ + _W = W + ix; + + /* inner loop */ + for (iy = 0; iy < n->used; iy++) { + *_W++ += ((mp_word)mu) * ((mp_word) * tmpn++); + } + } + + /* now fix carry for next digit, W[ix+1] */ + W[ix + 1] += W[ix] >> ((mp_word)DIGIT_BIT); + } + + /* now we have to propagate the carries and + * shift the words downward [all those least + * significant digits we zeroed]. + */ + { + mp_digit *tmpx; + mp_word *_W, *_W1; + + /* nox fix rest of carries */ + + /* alias for current word */ + _W1 = W + ix; + + /* alias for next word, where the carry goes */ + _W = W + ++ix; + + for ( ; ix <= ((n->used * 2) + 1); ix++) { + *_W++ += *_W1++ >> ((mp_word)DIGIT_BIT); + } + + /* copy out, A = A/b**n + * + * The result is A/b**n but instead of converting from an + * array of mp_word to mp_digit than calling mp_rshd + * we just copy them in the right order + */ + + /* alias for destination word */ + tmpx = x->dp; + + /* alias for shifted double precision result */ + _W = W + n->used; + + for (ix = 0; ix < (n->used + 1); ix++) { + *tmpx++ = (mp_digit)(*_W++ & ((mp_word)MP_MASK)); + } + + /* zero oldused digits, if the input a was larger than + * m->used+1 we'll have to clear the digits + */ + for ( ; ix < olduse; ix++) { + *tmpx++ = 0; + } + } + + /* set the max used and clamp */ + x->used = n->used + 1; + mp_clamp(x); + + /* if A >= m then A = A - m */ + if (mp_cmp_mag(x, n) != MP_LT) { + return s_mp_sub(x, n, x); + } + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_FAST_S_MP_MUL_DIGS_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* Fast (comba) multiplier + * + * This is the fast column-array [comba] multiplier. It is + * designed to compute the columns of the product first + * then handle the carries afterwards. This has the effect + * of making the nested loops that compute the columns very + * simple and schedulable on super-scalar processors. + * + * This has been modified to produce a variable number of + * digits of output so if say only a half-product is required + * you don't have to compute the upper half (a feature + * required for fast Barrett reduction). + * + * Based on Algorithm 14.12 on pp.595 of HAC. + * + */ +int fast_s_mp_mul_digs(mp_int *a, mp_int *b, mp_int *c, int digs) { + int olduse, res, pa, ix, iz; + mp_digit W[MP_WARRAY]; + mp_word _W; + + /* grow the destination as required */ + if (c->alloc < digs) { + if ((res = mp_grow(c, digs)) != MP_OKAY) { + return res; + } + } + + /* number of output digits to produce */ + pa = MIN(digs, a->used + b->used); + + /* clear the carry */ + _W = 0; + for (ix = 0; ix < pa; ix++) { + int tx, ty; + int iy; + mp_digit *tmpx, *tmpy; + + /* get offsets into the two bignums */ + ty = MIN(b->used - 1, ix); + tx = ix - ty; + + /* setup temp aliases */ + tmpx = a->dp + tx; + tmpy = b->dp + ty; + + /* this is the number of times the loop will iterrate, essentially + while (tx++ < a->used && ty-- >= 0) { ... } + */ + iy = MIN(a->used - tx, ty + 1); + + /* execute loop */ + for (iz = 0; iz < iy; ++iz) { + _W += ((mp_word) * tmpx++) * ((mp_word) * tmpy--); + } + + /* store term */ + W[ix] = ((mp_digit)_W) & MP_MASK; + + /* make next carry */ + _W = _W >> ((mp_word)DIGIT_BIT); + } + + /* setup dest */ + olduse = c->used; + c->used = pa; + + { + mp_digit *tmpc; + tmpc = c->dp; + for (ix = 0; ix < (pa + 1); ix++) { + /* now extract the previous digit [below the carry] */ + *tmpc++ = W[ix]; + } + + /* clear unused digits [that existed in the old copy of c] */ + for ( ; ix < olduse; ix++) { + *tmpc++ = 0; + } + } + mp_clamp(c); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_FAST_S_MP_MUL_HIGH_DIGS_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* this is a modified version of fast_s_mul_digs that only produces + * output digits *above* digs. See the comments for fast_s_mul_digs + * to see how it works. + * + * This is used in the Barrett reduction since for one of the multiplications + * only the higher digits were needed. This essentially halves the work. + * + * Based on Algorithm 14.12 on pp.595 of HAC. + */ +int fast_s_mp_mul_high_digs(mp_int *a, mp_int *b, mp_int *c, int digs) { + int olduse, res, pa, ix, iz; + mp_digit W[MP_WARRAY]; + mp_word _W; + + /* grow the destination as required */ + pa = a->used + b->used; + if (c->alloc < pa) { + if ((res = mp_grow(c, pa)) != MP_OKAY) { + return res; + } + } + + /* number of output digits to produce */ + pa = a->used + b->used; + _W = 0; + for (ix = digs; ix < pa; ix++) { + int tx, ty, iy; + mp_digit *tmpx, *tmpy; + + /* get offsets into the two bignums */ + ty = MIN(b->used - 1, ix); + tx = ix - ty; + + /* setup temp aliases */ + tmpx = a->dp + tx; + tmpy = b->dp + ty; + + /* this is the number of times the loop will iterrate, essentially its + while (tx++ < a->used && ty-- >= 0) { ... } + */ + iy = MIN(a->used - tx, ty + 1); + + /* execute loop */ + for (iz = 0; iz < iy; iz++) { + _W += ((mp_word) * tmpx++) * ((mp_word) * tmpy--); + } + + /* store term */ + W[ix] = ((mp_digit)_W) & MP_MASK; + + /* make next carry */ + _W = _W >> ((mp_word)DIGIT_BIT); + } + + /* setup dest */ + olduse = c->used; + c->used = pa; + + { + mp_digit *tmpc; + + tmpc = c->dp + digs; + for (ix = digs; ix < pa; ix++) { + /* now extract the previous digit [below the carry] */ + *tmpc++ = W[ix]; + } + + /* clear unused digits [that existed in the old copy of c] */ + for ( ; ix < olduse; ix++) { + *tmpc++ = 0; + } + } + mp_clamp(c); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_FAST_S_MP_SQR_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* the jist of squaring... + * you do like mult except the offset of the tmpx [one that + * starts closer to zero] can't equal the offset of tmpy. + * So basically you set up iy like before then you min it with + * (ty-tx) so that it never happens. You double all those + * you add in the inner loop + + After that loop you do the squares and add them in. + */ + +int fast_s_mp_sqr(mp_int *a, mp_int *b) { + int olduse, res, pa, ix, iz; + mp_digit W[MP_WARRAY], *tmpx; + mp_word W1; + + /* grow the destination as required */ + pa = a->used + a->used; + if (b->alloc < pa) { + if ((res = mp_grow(b, pa)) != MP_OKAY) { + return res; + } + } + + /* number of output digits to produce */ + W1 = 0; + for (ix = 0; ix < pa; ix++) { + int tx, ty, iy; + mp_word _W; + mp_digit *tmpy; + + /* clear counter */ + _W = 0; + + /* get offsets into the two bignums */ + ty = MIN(a->used - 1, ix); + tx = ix - ty; + + /* setup temp aliases */ + tmpx = a->dp + tx; + tmpy = a->dp + ty; + + /* this is the number of times the loop will iterrate, essentially + while (tx++ < a->used && ty-- >= 0) { ... } + */ + iy = MIN(a->used - tx, ty + 1); + + /* now for squaring tx can never equal ty + * we halve the distance since they approach at a rate of 2x + * and we have to round because odd cases need to be executed + */ + iy = MIN(iy, ((ty - tx) + 1) >> 1); + + /* execute loop */ + for (iz = 0; iz < iy; iz++) { + _W += ((mp_word) * tmpx++) * ((mp_word) * tmpy--); + } + + /* double the inner product and add carry */ + _W = _W + _W + W1; + + /* even columns have the square term in them */ + if ((ix & 1) == 0) { + _W += ((mp_word)a->dp[ix >> 1]) * ((mp_word)a->dp[ix >> 1]); + } + + /* store it */ + W[ix] = (mp_digit)(_W & MP_MASK); + + /* make next carry */ + W1 = _W >> ((mp_word)DIGIT_BIT); + } + + /* setup dest */ + olduse = b->used; + b->used = a->used + a->used; + + { + mp_digit *tmpb; + tmpb = b->dp; + for (ix = 0; ix < pa; ix++) { + *tmpb++ = W[ix] & MP_MASK; + } + + /* clear unused digits [that existed in the old copy of c] */ + for ( ; ix < olduse; ix++) { + *tmpb++ = 0; + } + } + mp_clamp(b); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_2EXPT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* computes a = 2**b + * + * Simple algorithm which zeroes the int, grows it then just sets one bit + * as required. + */ +int +mp_2expt(mp_int *a, int b) { + int res; + + /* zero a as per default */ + mp_zero(a); + + /* grow a to accomodate the single bit */ + if ((res = mp_grow(a, (b / DIGIT_BIT) + 1)) != MP_OKAY) { + return res; + } + + /* set the used count of where the bit will go */ + a->used = (b / DIGIT_BIT) + 1; + + /* put the single bit in its place */ + a->dp[b / DIGIT_BIT] = ((mp_digit)1) << (b % DIGIT_BIT); + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_ABS_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* b = |a| + * + * Simple function copies the input and fixes the sign to positive + */ +int +mp_abs(mp_int *a, mp_int *b) { + int res; + + /* copy a to b */ + if (a != b) { + if ((res = mp_copy(a, b)) != MP_OKAY) { + return res; + } + } + + /* force the sign of b to positive */ + b->sign = MP_ZPOS; + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_ADD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* high level addition (handles signs) */ +int mp_add(mp_int *a, mp_int *b, mp_int *c) { + int sa, sb, res; + + /* get sign of both inputs */ + sa = a->sign; + sb = b->sign; + + /* handle two cases, not four */ + if (sa == sb) { + /* both positive or both negative */ + /* add their magnitudes, copy the sign */ + c->sign = sa; + res = s_mp_add(a, b, c); + } else { + /* one positive, the other negative */ + /* subtract the one with the greater magnitude from */ + /* the one of the lesser magnitude. The result gets */ + /* the sign of the one with the greater magnitude. */ + if (mp_cmp_mag(a, b) == MP_LT) { + c->sign = sb; + res = s_mp_sub(b, a, c); + } else { + c->sign = sa; + res = s_mp_sub(a, b, c); + } + } + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_ADD_D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* single digit addition */ +int +mp_add_d(mp_int *a, mp_digit b, mp_int *c) { + int res, ix, oldused; + mp_digit *tmpa, *tmpc, mu; + + /* grow c as required */ + if (c->alloc < (a->used + 1)) { + if ((res = mp_grow(c, a->used + 1)) != MP_OKAY) { + return res; + } + } + + /* if a is negative and |a| >= b, call c = |a| - b */ + if ((a->sign == MP_NEG) && ((a->used > 1) || (a->dp[0] >= b))) { + /* temporarily fix sign of a */ + a->sign = MP_ZPOS; + + /* c = |a| - b */ + res = mp_sub_d(a, b, c); + + /* fix sign */ + a->sign = c->sign = MP_NEG; + + /* clamp */ + mp_clamp(c); + + return res; + } + + /* old number of used digits in c */ + oldused = c->used; + + /* sign always positive */ + c->sign = MP_ZPOS; + + /* source alias */ + tmpa = a->dp; + + /* destination alias */ + tmpc = c->dp; + + /* if a is positive */ + if (a->sign == MP_ZPOS) { + /* add digit, after this we're propagating + * the carry. + */ + *tmpc = *tmpa++ + b; + mu = *tmpc >> DIGIT_BIT; + *tmpc++ &= MP_MASK; + + /* now handle rest of the digits */ + for (ix = 1; ix < a->used; ix++) { + *tmpc = *tmpa++ + mu; + mu = *tmpc >> DIGIT_BIT; + *tmpc++ &= MP_MASK; + } + /* set final carry */ + ix++; + *tmpc++ = mu; + + /* setup size */ + c->used = a->used + 1; + } else { + /* a was negative and |a| < b */ + c->used = 1; + + /* the result is a single digit */ + if (a->used == 1) { + *tmpc++ = b - a->dp[0]; + } else { + *tmpc++ = b; + } + + /* setup count so the clearing of oldused + * can fall through correctly + */ + ix = 1; + } + + /* now zero to oldused */ + while (ix++ < oldused) { + *tmpc++ = 0; + } + mp_clamp(c); + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_ADDMOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* d = a + b (mod c) */ +int +mp_addmod(mp_int *a, mp_int *b, mp_int *c, mp_int *d) { + int res; + mp_int t; + + if ((res = mp_init(&t)) != MP_OKAY) { + return res; + } + + if ((res = mp_add(a, b, &t)) != MP_OKAY) { + mp_clear(&t); + return res; + } + res = mp_mod(&t, c, d); + mp_clear(&t); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_AND_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* AND two ints together */ +int +mp_and(mp_int *a, mp_int *b, mp_int *c) { + int res, ix, px; + mp_int t, *x; + + if (a->used > b->used) { + if ((res = mp_init_copy(&t, a)) != MP_OKAY) { + return res; + } + px = b->used; + x = b; + } else { + if ((res = mp_init_copy(&t, b)) != MP_OKAY) { + return res; + } + px = a->used; + x = a; + } + + for (ix = 0; ix < px; ix++) { + t.dp[ix] &= x->dp[ix]; + } + + /* zero digits above the last from the smallest mp_int */ + for ( ; ix < t.used; ix++) { + t.dp[ix] = 0; + } + + mp_clamp(&t); + mp_exch(c, &t); + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_CLAMP_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* trim unused digits + * + * This is used to ensure that leading zero digits are + * trimed and the leading "used" digit will be non-zero + * Typically very fast. Also fixes the sign if there + * are no more leading digits + */ +void +mp_clamp(mp_int *a) { + /* decrease used while the most significant digit is + * zero. + */ + while ((a->used > 0) && (a->dp[a->used - 1] == 0)) { + --(a->used); + } + + /* reset the sign flag if used == 0 */ + if (a->used == 0) { + a->sign = MP_ZPOS; + } +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_CLEAR_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* clear one (frees) */ +void +mp_clear(mp_int *a) { + int i; + + /* only do anything if a hasn't been freed previously */ + if (a->dp != NULL) { + /* first zero the digits */ + for (i = 0; i < a->used; i++) { + a->dp[i] = 0; + } + + /* free ram */ + XFREE(a->dp); + + /* reset members to make debugging easier */ + a->dp = NULL; + a->alloc = a->used = 0; + a->sign = MP_ZPOS; + } +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_CLEAR_MULTI_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ +#include + +void mp_clear_multi(mp_int *mp, ...) { + mp_int *next_mp = mp; + va_list args; + + va_start(args, mp); + while (next_mp != NULL) { + mp_clear(next_mp); + next_mp = va_arg(args, mp_int *); + } + va_end(args); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_CMP_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* compare two ints (signed)*/ +int +mp_cmp(mp_int *a, mp_int *b) { + /* compare based on sign */ + if (a->sign != b->sign) { + if (a->sign == MP_NEG) { + return MP_LT; + } else { + return MP_GT; + } + } + + /* compare digits */ + if (a->sign == MP_NEG) { + /* if negative compare opposite direction */ + return mp_cmp_mag(b, a); + } else { + return mp_cmp_mag(a, b); + } +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_CMP_D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* compare a digit */ +int mp_cmp_d(mp_int *a, mp_digit b) { + /* compare based on sign */ + if (a->sign == MP_NEG) { + return MP_LT; + } + + /* compare based on magnitude */ + if (a->used > 1) { + return MP_GT; + } + + /* compare the only digit of a to b */ + if (a->dp[0] > b) { + return MP_GT; + } else if (a->dp[0] < b) { + return MP_LT; + } else { + return MP_EQ; + } +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_CMP_MAG_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* compare maginitude of two ints (unsigned) */ +int mp_cmp_mag(mp_int *a, mp_int *b) { + int n; + mp_digit *tmpa, *tmpb; + + /* compare based on # of non-zero digits */ + if (a->used > b->used) { + return MP_GT; + } + + if (a->used < b->used) { + return MP_LT; + } + + /* alias for a */ + tmpa = a->dp + (a->used - 1); + + /* alias for b */ + tmpb = b->dp + (a->used - 1); + + /* compare based on digits */ + for (n = 0; n < a->used; ++n, --tmpa, --tmpb) { + if (*tmpa > *tmpb) { + return MP_GT; + } + + if (*tmpa < *tmpb) { + return MP_LT; + } + } + return MP_EQ; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_CNT_LSB_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +static const int lnz[16] = { + 4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0 +}; + +/* Counts the number of lsbs which are zero before the first zero bit */ +int mp_cnt_lsb(mp_int *a) { + int x; + mp_digit q, qq; + + /* easy out */ + if (mp_iszero(a) == MP_YES) { + return 0; + } + + /* scan lower digits until non-zero */ + for (x = 0; (x < a->used) && (a->dp[x] == 0); x++) { + } + q = a->dp[x]; + x *= DIGIT_BIT; + + /* now scan this digit until a 1 is found */ + if ((q & 1) == 0) { + do { + qq = q & 15; + x += lnz[qq]; + q >>= 4; + } while (qq == 0); + } + return x; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_COPY_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* copy, b = a */ +int +mp_copy(mp_int *a, mp_int *b) { + int res, n; + + /* if dst == src do nothing */ + if (a == b) { + return MP_OKAY; + } + + /* grow dest */ + if (b->alloc < a->used) { + if ((res = mp_grow(b, a->used)) != MP_OKAY) { + return res; + } + } + + /* zero b and copy the parameters over */ + { + mp_digit *tmpa, *tmpb; + + /* pointer aliases */ + + /* source */ + tmpa = a->dp; + + /* destination */ + tmpb = b->dp; + + /* copy all the digits */ + for (n = 0; n < a->used; n++) { + *tmpb++ = *tmpa++; + } + + /* clear high digits */ + for ( ; n < b->used; n++) { + *tmpb++ = 0; + } + } + + /* copy used count and sign */ + b->used = a->used; + b->sign = a->sign; + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_COUNT_BITS_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* returns the number of bits in an int */ +int +mp_count_bits(mp_int *a) { + int r; + mp_digit q; + + /* shortcut */ + if (a->used == 0) { + return 0; + } + + /* get number of digits and add that */ + r = (a->used - 1) * DIGIT_BIT; + + /* take the last digit and count the bits in it */ + q = a->dp[a->used - 1]; + while (q > ((mp_digit)0)) { + ++r; + q >>= ((mp_digit)1); + } + return r; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_DIV_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + + #ifdef BN_MP_DIV_SMALL + +/* slower bit-bang division... also smaller */ +int mp_div(mp_int *a, mp_int *b, mp_int *c, mp_int *d) { + mp_int ta, tb, tq, q; + int res, n, n2; + + /* is divisor zero ? */ + if (mp_iszero(b) == MP_YES) { + return MP_VAL; + } + + /* if a < b then q=0, r = a */ + if (mp_cmp_mag(a, b) == MP_LT) { + if (d != NULL) { + res = mp_copy(a, d); + } else { + res = MP_OKAY; + } + if (c != NULL) { + mp_zero(c); + } + return res; + } + + /* init our temps */ + if ((res = mp_init_multi(&ta, &tb, &tq, &q, NULL)) != MP_OKAY) { + return res; + } + + + mp_set(&tq, 1); + n = mp_count_bits(a) - mp_count_bits(b); + if (((res = mp_abs(a, &ta)) != MP_OKAY) || + ((res = mp_abs(b, &tb)) != MP_OKAY) || + ((res = mp_mul_2d(&tb, n, &tb)) != MP_OKAY) || + ((res = mp_mul_2d(&tq, n, &tq)) != MP_OKAY)) { + goto LBL_ERR; + } + + while (n-- >= 0) { + if (mp_cmp(&tb, &ta) != MP_GT) { + if (((res = mp_sub(&ta, &tb, &ta)) != MP_OKAY) || + ((res = mp_add(&q, &tq, &q)) != MP_OKAY)) { + goto LBL_ERR; + } + } + if (((res = mp_div_2d(&tb, 1, &tb, NULL)) != MP_OKAY) || + ((res = mp_div_2d(&tq, 1, &tq, NULL)) != MP_OKAY)) { + goto LBL_ERR; + } + } + + /* now q == quotient and ta == remainder */ + n = a->sign; + n2 = (a->sign == b->sign) ? MP_ZPOS : MP_NEG; + if (c != NULL) { + mp_exch(c, &q); + c->sign = (mp_iszero(c) == MP_YES) ? MP_ZPOS : n2; + } + if (d != NULL) { + mp_exch(d, &ta); + d->sign = (mp_iszero(d) == MP_YES) ? MP_ZPOS : n; + } +LBL_ERR: + mp_clear_multi(&ta, &tb, &tq, &q, NULL); + return res; +} + + #else + +/* integer signed division. + * c*b + d == a [e.g. a/b, c=quotient, d=remainder] + * HAC pp.598 Algorithm 14.20 + * + * Note that the description in HAC is horribly + * incomplete. For example, it doesn't consider + * the case where digits are removed from 'x' in + * the inner loop. It also doesn't consider the + * case that y has fewer than three digits, etc.. + * + * The overall algorithm is as described as + * 14.20 from HAC but fixed to treat these cases. + */ +int mp_div(mp_int *a, mp_int *b, mp_int *c, mp_int *d) { + mp_int q, x, y, t1, t2; + int res, n, t, i, norm, neg; + + /* is divisor zero ? */ + if (mp_iszero(b) == MP_YES) { + return MP_VAL; + } + + /* if a < b then q=0, r = a */ + if (mp_cmp_mag(a, b) == MP_LT) { + if (d != NULL) { + res = mp_copy(a, d); + } else { + res = MP_OKAY; + } + if (c != NULL) { + mp_zero(c); + } + return res; + } + + if ((res = mp_init_size(&q, a->used + 2)) != MP_OKAY) { + return res; + } + q.used = a->used + 2; + + if ((res = mp_init(&t1)) != MP_OKAY) { + goto LBL_Q; + } + + if ((res = mp_init(&t2)) != MP_OKAY) { + goto LBL_T1; + } + + if ((res = mp_init_copy(&x, a)) != MP_OKAY) { + goto LBL_T2; + } + + if ((res = mp_init_copy(&y, b)) != MP_OKAY) { + goto LBL_X; + } + + /* fix the sign */ + neg = (a->sign == b->sign) ? MP_ZPOS : MP_NEG; + x.sign = y.sign = MP_ZPOS; + + /* normalize both x and y, ensure that y >= b/2, [b == 2**DIGIT_BIT] */ + norm = mp_count_bits(&y) % DIGIT_BIT; + if (norm < (int)(DIGIT_BIT - 1)) { + norm = (DIGIT_BIT - 1) - norm; + if ((res = mp_mul_2d(&x, norm, &x)) != MP_OKAY) { + goto LBL_Y; + } + if ((res = mp_mul_2d(&y, norm, &y)) != MP_OKAY) { + goto LBL_Y; + } + } else { + norm = 0; + } + + /* note hac does 0 based, so if used==5 then its 0,1,2,3,4, e.g. use 4 */ + n = x.used - 1; + t = y.used - 1; + + /* while (x >= y*b**n-t) do { q[n-t] += 1; x -= y*b**{n-t} } */ + if ((res = mp_lshd(&y, n - t)) != MP_OKAY) { /* y = y*b**{n-t} */ + goto LBL_Y; + } + + while (mp_cmp(&x, &y) != MP_LT) { + ++(q.dp[n - t]); + if ((res = mp_sub(&x, &y, &x)) != MP_OKAY) { + goto LBL_Y; + } + } + + /* reset y by shifting it back down */ + mp_rshd(&y, n - t); + + /* step 3. for i from n down to (t + 1) */ + for (i = n; i >= (t + 1); i--) { + if (i > x.used) { + continue; + } + + /* step 3.1 if xi == yt then set q{i-t-1} to b-1, + * otherwise set q{i-t-1} to (xi*b + x{i-1})/yt */ + if (x.dp[i] == y.dp[t]) { + q.dp[(i - t) - 1] = ((((mp_digit)1) << DIGIT_BIT) - 1); + } else { + mp_word tmp; + tmp = ((mp_word)x.dp[i]) << ((mp_word)DIGIT_BIT); + tmp |= ((mp_word)x.dp[i - 1]); + tmp /= ((mp_word)y.dp[t]); + if (tmp > (mp_word)MP_MASK) { + tmp = MP_MASK; + } + q.dp[(i - t) - 1] = (mp_digit)(tmp & (mp_word)(MP_MASK)); + } + + /* while (q{i-t-1} * (yt * b + y{t-1})) > + xi * b**2 + xi-1 * b + xi-2 + + do q{i-t-1} -= 1; + */ + q.dp[(i - t) - 1] = (q.dp[(i - t) - 1] + 1) & MP_MASK; + do { + q.dp[(i - t) - 1] = (q.dp[(i - t) - 1] - 1) & MP_MASK; + + /* find left hand */ + mp_zero(&t1); + t1.dp[0] = ((t - 1) < 0) ? 0 : y.dp[t - 1]; + t1.dp[1] = y.dp[t]; + t1.used = 2; + if ((res = mp_mul_d(&t1, q.dp[(i - t) - 1], &t1)) != MP_OKAY) { + goto LBL_Y; + } + + /* find right hand */ + t2.dp[0] = ((i - 2) < 0) ? 0 : x.dp[i - 2]; + t2.dp[1] = ((i - 1) < 0) ? 0 : x.dp[i - 1]; + t2.dp[2] = x.dp[i]; + t2.used = 3; + } while (mp_cmp_mag(&t1, &t2) == MP_GT); + + /* step 3.3 x = x - q{i-t-1} * y * b**{i-t-1} */ + if ((res = mp_mul_d(&y, q.dp[(i - t) - 1], &t1)) != MP_OKAY) { + goto LBL_Y; + } + + if ((res = mp_lshd(&t1, (i - t) - 1)) != MP_OKAY) { + goto LBL_Y; + } + + if ((res = mp_sub(&x, &t1, &x)) != MP_OKAY) { + goto LBL_Y; + } + + /* if x < 0 then { x = x + y*b**{i-t-1}; q{i-t-1} -= 1; } */ + if (x.sign == MP_NEG) { + if ((res = mp_copy(&y, &t1)) != MP_OKAY) { + goto LBL_Y; + } + if ((res = mp_lshd(&t1, (i - t) - 1)) != MP_OKAY) { + goto LBL_Y; + } + if ((res = mp_add(&x, &t1, &x)) != MP_OKAY) { + goto LBL_Y; + } + + q.dp[(i - t) - 1] = (q.dp[(i - t) - 1] - 1UL) & MP_MASK; + } + } + + /* now q is the quotient and x is the remainder + * [which we have to normalize] + */ + + /* get sign before writing to c */ + x.sign = (x.used == 0) ? MP_ZPOS : a->sign; + + if (c != NULL) { + mp_clamp(&q); + mp_exch(&q, c); + c->sign = neg; + } + + if (d != NULL) { + if ((res = mp_div_2d(&x, norm, &x, NULL)) != MP_OKAY) { + goto LBL_Y; + } + mp_exch(&x, d); + } + + res = MP_OKAY; + +LBL_Y: mp_clear(&y); +LBL_X: mp_clear(&x); +LBL_T2: mp_clear(&t2); +LBL_T1: mp_clear(&t1); +LBL_Q: mp_clear(&q); + return res; +} + #endif +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_DIV_2_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* b = a/2 */ +int mp_div_2(mp_int *a, mp_int *b) { + int x, res, oldused; + + /* copy */ + if (b->alloc < a->used) { + if ((res = mp_grow(b, a->used)) != MP_OKAY) { + return res; + } + } + + oldused = b->used; + b->used = a->used; + { + mp_digit r, rr, *tmpa, *tmpb; + + /* source alias */ + tmpa = a->dp + b->used - 1; + + /* dest alias */ + tmpb = b->dp + b->used - 1; + + /* carry */ + r = 0; + for (x = b->used - 1; x >= 0; x--) { + /* get the carry for the next iteration */ + rr = *tmpa & 1; + + /* shift the current digit, add in carry and store */ + *tmpb-- = (*tmpa-- >> 1) | (r << (DIGIT_BIT - 1)); + + /* forward carry to next iteration */ + r = rr; + } + + /* zero excess digits */ + tmpb = b->dp + b->used; + for (x = b->used; x < oldused; x++) { + *tmpb++ = 0; + } + } + b->sign = a->sign; + mp_clamp(b); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_DIV_2D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* shift right by a certain bit count (store quotient in c, optional remainder in d) */ +int mp_div_2d(mp_int *a, int b, mp_int *c, mp_int *d) { + mp_digit D, r, rr; + int x, res; + mp_int t; + + + /* if the shift count is <= 0 then we do no work */ + if (b <= 0) { + res = mp_copy(a, c); + if (d != NULL) { + mp_zero(d); + } + return res; + } + + if ((res = mp_init(&t)) != MP_OKAY) { + return res; + } + + /* get the remainder */ + if (d != NULL) { + if ((res = mp_mod_2d(a, b, &t)) != MP_OKAY) { + mp_clear(&t); + return res; + } + } + + /* copy */ + if ((res = mp_copy(a, c)) != MP_OKAY) { + mp_clear(&t); + return res; + } + + /* shift by as many digits in the bit count */ + if (b >= (int)DIGIT_BIT) { + mp_rshd(c, b / DIGIT_BIT); + } + + /* shift any bit count < DIGIT_BIT */ + D = (mp_digit)(b % DIGIT_BIT); + if (D != 0) { + mp_digit *tmpc, mask, shift; + + /* mask */ + mask = (((mp_digit)1) << D) - 1; + + /* shift for lsb */ + shift = DIGIT_BIT - D; + + /* alias */ + tmpc = c->dp + (c->used - 1); + + /* carry */ + r = 0; + for (x = c->used - 1; x >= 0; x--) { + /* get the lower bits of this word in a temp */ + rr = *tmpc & mask; + + /* shift the current word and mix in the carry bits from the previous word */ + *tmpc = (*tmpc >> D) | (r << shift); + --tmpc; + + /* set the carry to the carry bits of the current word found above */ + r = rr; + } + } + mp_clamp(c); + if (d != NULL) { + mp_exch(&t, d); + } + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_DIV_3_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* divide by three (based on routine from MPI and the GMP manual) */ +int +mp_div_3(mp_int *a, mp_int *c, mp_digit *d) { + mp_int q; + mp_word w, t; + mp_digit b; + int res, ix; + + /* b = 2**DIGIT_BIT / 3 */ + b = (((mp_word)1) << ((mp_word)DIGIT_BIT)) / ((mp_word)3); + + if ((res = mp_init_size(&q, a->used)) != MP_OKAY) { + return res; + } + + q.used = a->used; + q.sign = a->sign; + w = 0; + for (ix = a->used - 1; ix >= 0; ix--) { + w = (w << ((mp_word)DIGIT_BIT)) | ((mp_word)a->dp[ix]); + + if (w >= 3) { + /* multiply w by [1/3] */ + t = (w * ((mp_word)b)) >> ((mp_word)DIGIT_BIT); + + /* now subtract 3 * [w/3] from w, to get the remainder */ + w -= t + t + t; + + /* fixup the remainder as required since + * the optimization is not exact. + */ + while (w >= 3) { + t += 1; + w -= 3; + } + } else { + t = 0; + } + q.dp[ix] = (mp_digit)t; + } + + /* [optional] store the remainder */ + if (d != NULL) { + *d = (mp_digit)w; + } + + /* [optional] store the quotient */ + if (c != NULL) { + mp_clamp(&q); + mp_exch(&q, c); + } + mp_clear(&q); + + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_DIV_D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +static int s_is_power_of_two(mp_digit b, int *p) { + int x; + + /* fast return if no power of two */ + if ((b == 0) || ((b & (b - 1)) != 0)) { + return 0; + } + + for (x = 0; x < DIGIT_BIT; x++) { + if (b == (((mp_digit)1) << x)) { + *p = x; + return 1; + } + } + return 0; +} + +/* single digit division (based on routine from MPI) */ +int mp_div_d(mp_int *a, mp_digit b, mp_int *c, mp_digit *d) { + mp_int q; + mp_word w; + mp_digit t; + int res, ix; + + /* cannot divide by zero */ + if (b == 0) { + return MP_VAL; + } + + /* quick outs */ + if ((b == 1) || (mp_iszero(a) == MP_YES)) { + if (d != NULL) { + *d = 0; + } + if (c != NULL) { + return mp_copy(a, c); + } + return MP_OKAY; + } + + /* power of two ? */ + if (s_is_power_of_two(b, &ix) == 1) { + if (d != NULL) { + *d = a->dp[0] & ((((mp_digit)1) << ix) - 1); + } + if (c != NULL) { + return mp_div_2d(a, ix, c, NULL); + } + return MP_OKAY; + } + + #ifdef BN_MP_DIV_3_C + /* three? */ + if (b == 3) { + return mp_div_3(a, c, d); + } + #endif + + /* no easy answer [c'est la vie]. Just division */ + if ((res = mp_init_size(&q, a->used)) != MP_OKAY) { + return res; + } + + q.used = a->used; + q.sign = a->sign; + w = 0; + for (ix = a->used - 1; ix >= 0; ix--) { + w = (w << ((mp_word)DIGIT_BIT)) | ((mp_word)a->dp[ix]); + + if (w >= b) { + t = (mp_digit)(w / b); + w -= ((mp_word)t) * ((mp_word)b); + } else { + t = 0; + } + q.dp[ix] = (mp_digit)t; + } + + if (d != NULL) { + *d = (mp_digit)w; + } + + if (c != NULL) { + mp_clamp(&q); + mp_exch(&q, c); + } + mp_clear(&q); + + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_DR_IS_MODULUS_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* determines if a number is a valid DR modulus */ +int mp_dr_is_modulus(mp_int *a) { + int ix; + + /* must be at least two digits */ + if (a->used < 2) { + return 0; + } + + /* must be of the form b**k - a [a <= b] so all + * but the first digit must be equal to -1 (mod b). + */ + for (ix = 1; ix < a->used; ix++) { + if (a->dp[ix] != MP_MASK) { + return 0; + } + } + return 1; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_DR_REDUCE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* reduce "x" in place modulo "n" using the Diminished Radix algorithm. + * + * Based on algorithm from the paper + * + * "Generating Efficient Primes for Discrete Log Cryptosystems" + * Chae Hoon Lim, Pil Joong Lee, + * POSTECH Information Research Laboratories + * + * The modulus must be of a special format [see manual] + * + * Has been modified to use algorithm 7.10 from the LTM book instead + * + * Input x must be in the range 0 <= x <= (n-1)**2 + */ +int +mp_dr_reduce(mp_int *x, mp_int *n, mp_digit k) { + int err, i, m; + mp_word r; + mp_digit mu, *tmpx1, *tmpx2; + + /* m = digits in modulus */ + m = n->used; + + /* ensure that "x" has at least 2m digits */ + if (x->alloc < (m + m)) { + if ((err = mp_grow(x, m + m)) != MP_OKAY) { + return err; + } + } + +/* top of loop, this is where the code resumes if + * another reduction pass is required. + */ +top: + /* aliases for digits */ + /* alias for lower half of x */ + tmpx1 = x->dp; + + /* alias for upper half of x, or x/B**m */ + tmpx2 = x->dp + m; + + /* set carry to zero */ + mu = 0; + + /* compute (x mod B**m) + k * [x/B**m] inline and inplace */ + for (i = 0; i < m; i++) { + r = (((mp_word) * tmpx2++) * (mp_word)k) + *tmpx1 + mu; + *tmpx1++ = (mp_digit)(r & MP_MASK); + mu = (mp_digit)(r >> ((mp_word)DIGIT_BIT)); + } + + /* set final carry */ + *tmpx1++ = mu; + + /* zero words above m */ + for (i = m + 1; i < x->used; i++) { + *tmpx1++ = 0; + } + + /* clamp, sub and return */ + mp_clamp(x); + + /* if x >= n then subtract and reduce again + * Each successive "recursion" makes the input smaller and smaller. + */ + if (mp_cmp_mag(x, n) != MP_LT) { + if ((err = s_mp_sub(x, n, x)) != MP_OKAY) { + return err; + } + goto top; + } + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_DR_SETUP_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* determines the setup value */ +void mp_dr_setup(mp_int *a, mp_digit *d) { + /* the casts are required if DIGIT_BIT is one less than + * the number of bits in a mp_digit [e.g. DIGIT_BIT==31] + */ + *d = (mp_digit)((((mp_word)1) << ((mp_word)DIGIT_BIT)) - + ((mp_word)a->dp[0])); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_EXCH_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* swap the elements of two integers, for cases where you can't simply swap the + * mp_int pointers around + */ +void +mp_exch(mp_int *a, mp_int *b) { + mp_int t; + + t = *a; + *a = *b; + *b = t; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_EXPORT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* based on gmp's mpz_export. + * see http://gmplib.org/manual/Integer-Import-and-Export.html + */ +int mp_export(void *rop, size_t *countp, int order, size_t size, + int endian, size_t nails, mp_int *op) { + int result; + size_t odd_nails, nail_bytes, i, j, bits, count; + unsigned char odd_nail_mask; + + mp_int t; + + if ((result = mp_init_copy(&t, op)) != MP_OKAY) { + return result; + } + + if (endian == 0) { + union { + unsigned int i; + char c[4]; + } lint; + lint.i = 0x01020304; + + endian = (lint.c[0] == 4) ? -1 : 1; + } + + odd_nails = (nails % 8); + odd_nail_mask = 0xff; + for (i = 0; i < odd_nails; ++i) { + odd_nail_mask ^= (1 << (7 - i)); + } + nail_bytes = nails / 8; + + bits = mp_count_bits(&t); + count = (bits / ((size * 8) - nails)) + (((bits % ((size * 8) - nails)) != 0) ? 1 : 0); + + for (i = 0; i < count; ++i) { + for (j = 0; j < size; ++j) { + unsigned char *byte = ( + (unsigned char *)rop + + (((order == -1) ? i : ((count - 1) - i)) * size) + + ((endian == -1) ? j : ((size - 1) - j)) + ); + + if (j >= (size - nail_bytes)) { + *byte = 0; + continue; + } + + *byte = (unsigned char)((j == ((size - nail_bytes) - 1)) ? (t.dp[0] & odd_nail_mask) : (t.dp[0] & 0xFF)); + + if ((result = mp_div_2d(&t, ((j == ((size - nail_bytes) - 1)) ? (8 - odd_nails) : 8), &t, NULL)) != MP_OKAY) { + mp_clear(&t); + return result; + } + } + } + + mp_clear(&t); + + if (countp != NULL) { + *countp = count; + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_EXPT_D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* wrapper function for mp_expt_d_ex() */ +int mp_expt_d(mp_int *a, mp_digit b, mp_int *c) { + return mp_expt_d_ex(a, b, c, 0); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_EXPT_D_EX_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* calculate c = a**b using a square-multiply algorithm */ +int mp_expt_d_ex(mp_int *a, mp_digit b, mp_int *c, int fast) { + int res; + unsigned int x; + + mp_int g; + + if ((res = mp_init_copy(&g, a)) != MP_OKAY) { + return res; + } + + /* set initial result */ + mp_set(c, 1); + + if (fast != 0) { + while (b > 0) { + /* if the bit is set multiply */ + if ((b & 1) != 0) { + if ((res = mp_mul(c, &g, c)) != MP_OKAY) { + mp_clear(&g); + return res; + } + } + + /* square */ + if (b > 1) { + if ((res = mp_sqr(&g, &g)) != MP_OKAY) { + mp_clear(&g); + return res; + } + } + + /* shift to next bit */ + b >>= 1; + } + } else { + for (x = 0; x < DIGIT_BIT; x++) { + /* square */ + if ((res = mp_sqr(c, c)) != MP_OKAY) { + mp_clear(&g); + return res; + } + + /* if the bit is set multiply */ + if ((b & (mp_digit)(((mp_digit)1) << (DIGIT_BIT - 1))) != 0) { + if ((res = mp_mul(c, &g, c)) != MP_OKAY) { + mp_clear(&g); + return res; + } + } + + /* shift to next bit */ + b <<= 1; + } + } /* if ... else */ + + mp_clear(&g); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_EXPTMOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + + +/* this is a shell function that calls either the normal or Montgomery + * exptmod functions. Originally the call to the montgomery code was + * embedded in the normal function but that wasted alot of stack space + * for nothing (since 99% of the time the Montgomery code would be called) + */ +int mp_exptmod(mp_int *G, mp_int *X, mp_int *P, mp_int *Y) { + int dr; + + /* modulus P must be positive */ + if (P->sign == MP_NEG) { + return MP_VAL; + } + + /* if exponent X is negative we have to recurse */ + if (X->sign == MP_NEG) { + #ifdef BN_MP_INVMOD_C + mp_int tmpG, tmpX; + int err; + + /* first compute 1/G mod P */ + if ((err = mp_init(&tmpG)) != MP_OKAY) { + return err; + } + if ((err = mp_invmod(G, P, &tmpG)) != MP_OKAY) { + mp_clear(&tmpG); + return err; + } + + /* now get |X| */ + if ((err = mp_init(&tmpX)) != MP_OKAY) { + mp_clear(&tmpG); + return err; + } + if ((err = mp_abs(X, &tmpX)) != MP_OKAY) { + mp_clear_multi(&tmpG, &tmpX, NULL); + return err; + } + + /* and now compute (1/G)**|X| instead of G**X [X < 0] */ + err = mp_exptmod(&tmpG, &tmpX, P, Y); + mp_clear_multi(&tmpG, &tmpX, NULL); + return err; + #else + /* no invmod */ + return MP_VAL; + #endif + } + +/* modified diminished radix reduction */ + #if defined(BN_MP_REDUCE_IS_2K_L_C) && defined(BN_MP_REDUCE_2K_L_C) && defined(BN_S_MP_EXPTMOD_C) + if (mp_reduce_is_2k_l(P) == MP_YES) { + return s_mp_exptmod(G, X, P, Y, 1); + } + #endif + + #ifdef BN_MP_DR_IS_MODULUS_C + /* is it a DR modulus? */ + dr = mp_dr_is_modulus(P); + #else + /* default to no */ + dr = 0; + #endif + + #ifdef BN_MP_REDUCE_IS_2K_C + /* if not, is it a unrestricted DR modulus? */ + if (dr == 0) { + dr = mp_reduce_is_2k(P) << 1; + } + #endif + + /* if the modulus is odd or dr != 0 use the montgomery method */ + #ifdef BN_MP_EXPTMOD_FAST_C + if ((mp_isodd(P) == MP_YES) || (dr != 0)) { + return mp_exptmod_fast(G, X, P, Y, dr); + } else { + #endif + #ifdef BN_S_MP_EXPTMOD_C + /* otherwise use the generic Barrett reduction technique */ + return s_mp_exptmod(G, X, P, Y, 0); + #else + /* no exptmod for evens */ + return MP_VAL; + #endif + #ifdef BN_MP_EXPTMOD_FAST_C +} + #endif +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_EXPTMOD_FAST_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* computes Y == G**X mod P, HAC pp.616, Algorithm 14.85 + * + * Uses a left-to-right k-ary sliding window to compute the modular exponentiation. + * The value of k changes based on the size of the exponent. + * + * Uses Montgomery or Diminished Radix reduction [whichever appropriate] + */ + + #ifdef MP_LOW_MEM + #define TAB_SIZE 32 + #else + #define TAB_SIZE 256 + #endif + +int mp_exptmod_fast(mp_int *G, mp_int *X, mp_int *P, mp_int *Y, int redmode) { + mp_int M[TAB_SIZE], res; + mp_digit buf, mp; + int err, bitbuf, bitcpy, bitcnt, mode, digidx, x, y, winsize; + + /* use a pointer to the reduction algorithm. This allows us to use + * one of many reduction algorithms without modding the guts of + * the code with if statements everywhere. + */ + int (*redux)(mp_int *, mp_int *, mp_digit); + + /* find window size */ + x = mp_count_bits(X); + if (x <= 7) { + winsize = 2; + } else if (x <= 36) { + winsize = 3; + } else if (x <= 140) { + winsize = 4; + } else if (x <= 450) { + winsize = 5; + } else if (x <= 1303) { + winsize = 6; + } else if (x <= 3529) { + winsize = 7; + } else { + winsize = 8; + } + + #ifdef MP_LOW_MEM + if (winsize > 5) { + winsize = 5; + } + #endif + + /* init M array */ + /* init first cell */ + if ((err = mp_init(&M[1])) != MP_OKAY) { + return err; + } + + /* now init the second half of the array */ + for (x = 1 << (winsize - 1); x < (1 << winsize); x++) { + if ((err = mp_init(&M[x])) != MP_OKAY) { + for (y = 1 << (winsize - 1); y < x; y++) { + mp_clear(&M[y]); + } + mp_clear(&M[1]); + return err; + } + } + + /* determine and setup reduction code */ + if (redmode == 0) { + #ifdef BN_MP_MONTGOMERY_SETUP_C + /* now setup montgomery */ + if ((err = mp_montgomery_setup(P, &mp)) != MP_OKAY) { + goto LBL_M; + } + #else + err = MP_VAL; + goto LBL_M; + #endif + + /* automatically pick the comba one if available (saves quite a few calls/ifs) */ + #ifdef BN_FAST_MP_MONTGOMERY_REDUCE_C + if ((((P->used * 2) + 1) < MP_WARRAY) && + (P->used < (1 << ((CHAR_BIT * sizeof(mp_word)) - (2 * DIGIT_BIT))))) { + redux = fast_mp_montgomery_reduce; + } else + #endif + { + #ifdef BN_MP_MONTGOMERY_REDUCE_C + /* use slower baseline Montgomery method */ + redux = mp_montgomery_reduce; + #else + err = MP_VAL; + goto LBL_M; + #endif + } + } else if (redmode == 1) { + #if defined(BN_MP_DR_SETUP_C) && defined(BN_MP_DR_REDUCE_C) + /* setup DR reduction for moduli of the form B**k - b */ + mp_dr_setup(P, &mp); + redux = mp_dr_reduce; + #else + err = MP_VAL; + goto LBL_M; + #endif + } else { + #if defined(BN_MP_REDUCE_2K_SETUP_C) && defined(BN_MP_REDUCE_2K_C) + /* setup DR reduction for moduli of the form 2**k - b */ + if ((err = mp_reduce_2k_setup(P, &mp)) != MP_OKAY) { + goto LBL_M; + } + redux = mp_reduce_2k; + #else + err = MP_VAL; + goto LBL_M; + #endif + } + + /* setup result */ + if ((err = mp_init(&res)) != MP_OKAY) { + goto LBL_M; + } + + /* create M table + * + + * + * The first half of the table is not computed though accept for M[0] and M[1] + */ + + if (redmode == 0) { + #ifdef BN_MP_MONTGOMERY_CALC_NORMALIZATION_C + /* now we need R mod m */ + if ((err = mp_montgomery_calc_normalization(&res, P)) != MP_OKAY) { + goto LBL_RES; + } + #else + err = MP_VAL; + goto LBL_RES; + #endif + + /* now set M[1] to G * R mod m */ + if ((err = mp_mulmod(G, &res, P, &M[1])) != MP_OKAY) { + goto LBL_RES; + } + } else { + mp_set(&res, 1); + if ((err = mp_mod(G, P, &M[1])) != MP_OKAY) { + goto LBL_RES; + } + } + + /* compute the value at M[1<<(winsize-1)] by squaring M[1] (winsize-1) times */ + if ((err = mp_copy(&M[1], &M[1 << (winsize - 1)])) != MP_OKAY) { + goto LBL_RES; + } + + for (x = 0; x < (winsize - 1); x++) { + if ((err = mp_sqr(&M[1 << (winsize - 1)], &M[1 << (winsize - 1)])) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&M[1 << (winsize - 1)], P, mp)) != MP_OKAY) { + goto LBL_RES; + } + } + + /* create upper table */ + for (x = (1 << (winsize - 1)) + 1; x < (1 << winsize); x++) { + if ((err = mp_mul(&M[x - 1], &M[1], &M[x])) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&M[x], P, mp)) != MP_OKAY) { + goto LBL_RES; + } + } + + /* set initial mode and bit cnt */ + mode = 0; + bitcnt = 1; + buf = 0; + digidx = X->used - 1; + bitcpy = 0; + bitbuf = 0; + + for ( ; ; ) { + /* grab next digit as required */ + if (--bitcnt == 0) { + /* if digidx == -1 we are out of digits so break */ + if (digidx == -1) { + break; + } + /* read next digit and reset bitcnt */ + buf = X->dp[digidx--]; + bitcnt = (int)DIGIT_BIT; + } + + /* grab the next msb from the exponent */ + y = (mp_digit)(buf >> (DIGIT_BIT - 1)) & 1; + buf <<= (mp_digit)1; + + /* if the bit is zero and mode == 0 then we ignore it + * These represent the leading zero bits before the first 1 bit + * in the exponent. Technically this opt is not required but it + * does lower the # of trivial squaring/reductions used + */ + if ((mode == 0) && (y == 0)) { + continue; + } + + /* if the bit is zero and mode == 1 then we square */ + if ((mode == 1) && (y == 0)) { + if ((err = mp_sqr(&res, &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, mp)) != MP_OKAY) { + goto LBL_RES; + } + continue; + } + + /* else we add it to the window */ + bitbuf |= (y << (winsize - ++bitcpy)); + mode = 2; + + if (bitcpy == winsize) { + /* ok window is filled so square as required and multiply */ + /* square first */ + for (x = 0; x < winsize; x++) { + if ((err = mp_sqr(&res, &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, mp)) != MP_OKAY) { + goto LBL_RES; + } + } + + /* then multiply */ + if ((err = mp_mul(&res, &M[bitbuf], &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, mp)) != MP_OKAY) { + goto LBL_RES; + } + + /* empty window and reset */ + bitcpy = 0; + bitbuf = 0; + mode = 1; + } + } + + /* if bits remain then square/multiply */ + if ((mode == 2) && (bitcpy > 0)) { + /* square then multiply if the bit is set */ + for (x = 0; x < bitcpy; x++) { + if ((err = mp_sqr(&res, &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, mp)) != MP_OKAY) { + goto LBL_RES; + } + + /* get next bit of the window */ + bitbuf <<= 1; + if ((bitbuf & (1 << winsize)) != 0) { + /* then multiply */ + if ((err = mp_mul(&res, &M[1], &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, mp)) != MP_OKAY) { + goto LBL_RES; + } + } + } + } + + if (redmode == 0) { + /* fixup result if Montgomery reduction is used + * recall that any value in a Montgomery system is + * actually multiplied by R mod n. So we have + * to reduce one more time to cancel out the factor + * of R. + */ + if ((err = redux(&res, P, mp)) != MP_OKAY) { + goto LBL_RES; + } + } + + /* swap res with Y */ + mp_exch(&res, Y); + err = MP_OKAY; +LBL_RES: mp_clear(&res); +LBL_M: + mp_clear(&M[1]); + for (x = 1 << (winsize - 1); x < (1 << winsize); x++) { + mp_clear(&M[x]); + } + return err; +} +#endif + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_EXTEUCLID_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* Extended euclidean algorithm of (a, b) produces + a*u1 + b*u2 = u3 + */ +int mp_exteuclid(mp_int *a, mp_int *b, mp_int *U1, mp_int *U2, mp_int *U3) { + mp_int u1, u2, u3, v1, v2, v3, t1, t2, t3, q, tmp; + int err; + + if ((err = mp_init_multi(&u1, &u2, &u3, &v1, &v2, &v3, &t1, &t2, &t3, &q, &tmp, NULL)) != MP_OKAY) { + return err; + } + + /* initialize, (u1,u2,u3) = (1,0,a) */ + mp_set(&u1, 1); + if ((err = mp_copy(a, &u3)) != MP_OKAY) { + goto _ERR; + } + + /* initialize, (v1,v2,v3) = (0,1,b) */ + mp_set(&v2, 1); + if ((err = mp_copy(b, &v3)) != MP_OKAY) { + goto _ERR; + } + + /* loop while v3 != 0 */ + while (mp_iszero(&v3) == MP_NO) { + /* q = u3/v3 */ + if ((err = mp_div(&u3, &v3, &q, NULL)) != MP_OKAY) { + goto _ERR; + } + + /* (t1,t2,t3) = (u1,u2,u3) - (v1,v2,v3)q */ + if ((err = mp_mul(&v1, &q, &tmp)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_sub(&u1, &tmp, &t1)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_mul(&v2, &q, &tmp)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_sub(&u2, &tmp, &t2)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_mul(&v3, &q, &tmp)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_sub(&u3, &tmp, &t3)) != MP_OKAY) { + goto _ERR; + } + + /* (u1,u2,u3) = (v1,v2,v3) */ + if ((err = mp_copy(&v1, &u1)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_copy(&v2, &u2)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_copy(&v3, &u3)) != MP_OKAY) { + goto _ERR; + } + + /* (v1,v2,v3) = (t1,t2,t3) */ + if ((err = mp_copy(&t1, &v1)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_copy(&t2, &v2)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_copy(&t3, &v3)) != MP_OKAY) { + goto _ERR; + } + } + + /* make sure U3 >= 0 */ + if (u3.sign == MP_NEG) { + if ((err = mp_neg(&u1, &u1)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_neg(&u2, &u2)) != MP_OKAY) { + goto _ERR; + } + if ((err = mp_neg(&u3, &u3)) != MP_OKAY) { + goto _ERR; + } + } + + /* copy result out */ + if (U1 != NULL) { + mp_exch(U1, &u1); + } + if (U2 != NULL) { + mp_exch(U2, &u2); + } + if (U3 != NULL) { + mp_exch(U3, &u3); + } + + err = MP_OKAY; +_ERR: mp_clear_multi(&u1, &u2, &u3, &v1, &v2, &v3, &t1, &t2, &t3, &q, &tmp, NULL); + return err; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_FREAD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* read a bigint from a file stream in ASCII */ +int mp_fread(mp_int *a, int radix, FILE *stream) { + int err, ch, neg, y; + + /* clear a */ + mp_zero(a); + + /* if first digit is - then set negative */ + ch = fgetc(stream); + if (ch == '-') { + neg = MP_NEG; + ch = fgetc(stream); + } else { + neg = MP_ZPOS; + } + + for ( ; ; ) { + /* find y in the radix map */ + for (y = 0; y < radix; y++) { + if (mp_s_rmap[y] == ch) { + break; + } + } + if (y == radix) { + break; + } + + /* shift up and add */ + if ((err = mp_mul_d(a, radix, a)) != MP_OKAY) { + return err; + } + if ((err = mp_add_d(a, y, a)) != MP_OKAY) { + return err; + } + + ch = fgetc(stream); + } + if (mp_cmp_d(a, 0) != MP_EQ) { + a->sign = neg; + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_FWRITE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +int mp_fwrite(mp_int *a, int radix, FILE *stream) { + char *buf; + int err, len, x; + + if ((err = mp_radix_size(a, radix, &len)) != MP_OKAY) { + return err; + } + + buf = OPT_CAST(char) XMALLOC(len); + if (buf == NULL) { + return MP_MEM; + } + + if ((err = mp_toradix(a, buf, radix)) != MP_OKAY) { + XFREE(buf); + return err; + } + + for (x = 0; x < len; x++) { + if (fputc(buf[x], stream) == EOF) { + XFREE(buf); + return MP_VAL; + } + } + + XFREE(buf); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_GCD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* Greatest Common Divisor using the binary method */ +int mp_gcd(mp_int *a, mp_int *b, mp_int *c) { + mp_int u, v; + int k, u_lsb, v_lsb, res; + + /* either zero than gcd is the largest */ + if (mp_iszero(a) == MP_YES) { + return mp_abs(b, c); + } + if (mp_iszero(b) == MP_YES) { + return mp_abs(a, c); + } + + /* get copies of a and b we can modify */ + if ((res = mp_init_copy(&u, a)) != MP_OKAY) { + return res; + } + + if ((res = mp_init_copy(&v, b)) != MP_OKAY) { + goto LBL_U; + } + + /* must be positive for the remainder of the algorithm */ + u.sign = v.sign = MP_ZPOS; + + /* B1. Find the common power of two for u and v */ + u_lsb = mp_cnt_lsb(&u); + v_lsb = mp_cnt_lsb(&v); + k = MIN(u_lsb, v_lsb); + + if (k > 0) { + /* divide the power of two out */ + if ((res = mp_div_2d(&u, k, &u, NULL)) != MP_OKAY) { + goto LBL_V; + } + + if ((res = mp_div_2d(&v, k, &v, NULL)) != MP_OKAY) { + goto LBL_V; + } + } + + /* divide any remaining factors of two out */ + if (u_lsb != k) { + if ((res = mp_div_2d(&u, u_lsb - k, &u, NULL)) != MP_OKAY) { + goto LBL_V; + } + } + + if (v_lsb != k) { + if ((res = mp_div_2d(&v, v_lsb - k, &v, NULL)) != MP_OKAY) { + goto LBL_V; + } + } + + while (mp_iszero(&v) == MP_NO) { + /* make sure v is the largest */ + if (mp_cmp_mag(&u, &v) == MP_GT) { + /* swap u and v to make sure v is >= u */ + mp_exch(&u, &v); + } + + /* subtract smallest from largest */ + if ((res = s_mp_sub(&v, &u, &v)) != MP_OKAY) { + goto LBL_V; + } + + /* Divide out all factors of two */ + if ((res = mp_div_2d(&v, mp_cnt_lsb(&v), &v, NULL)) != MP_OKAY) { + goto LBL_V; + } + } + + /* multiply by 2**k which we divided out at the beginning */ + if ((res = mp_mul_2d(&u, k, c)) != MP_OKAY) { + goto LBL_V; + } + c->sign = MP_ZPOS; + res = MP_OKAY; +LBL_V: mp_clear(&u); +LBL_U: mp_clear(&v); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_GET_INT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* get the lower 32-bits of an mp_int */ +unsigned long mp_get_int(mp_int *a) { + int i; + mp_min_u32 res; + + if (a->used == 0) { + return 0; + } + + /* get number of digits of the lsb we have to read */ + i = MIN(a->used, (int)(((sizeof(unsigned long) * CHAR_BIT) + DIGIT_BIT - 1) / DIGIT_BIT)) - 1; + + /* get most significant digit of result */ + res = DIGIT(a, i); + + while (--i >= 0) { + res = (res << DIGIT_BIT) | DIGIT(a, i); + } + + /* force result to 32-bits always so it is consistent on non 32-bit platforms */ + return res & 0xFFFFFFFFUL; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_GET_LONG_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* get the lower unsigned long of an mp_int, platform dependent */ +unsigned long mp_get_long(mp_int *a) { + int i; + unsigned long res; + + if (a->used == 0) { + return 0; + } + + /* get number of digits of the lsb we have to read */ + i = MIN(a->used, (int)(((sizeof(unsigned long) * CHAR_BIT) + DIGIT_BIT - 1) / DIGIT_BIT)) - 1; + + /* get most significant digit of result */ + res = DIGIT(a, i); + + #if (ULONG_MAX != 0xffffffffuL) || (DIGIT_BIT < 32) + while (--i >= 0) { + res = (res << DIGIT_BIT) | DIGIT(a, i); + } + #endif + return res; +} +#endif + + + +#ifdef BN_MP_GET_LONG_LONG_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* get the lower unsigned long long of an mp_int, platform dependent */ +unsigned long long mp_get_long_long(mp_int *a) { + int i; + unsigned long long res; + + if (a->used == 0) { + return 0; + } + + /* get number of digits of the lsb we have to read */ + i = MIN(a->used, (int)(((sizeof(unsigned long long) * CHAR_BIT) + DIGIT_BIT - 1) / DIGIT_BIT)) - 1; + + /* get most significant digit of result */ + res = DIGIT(a, i); + + #if DIGIT_BIT < 64 + while (--i >= 0) { + res = (res << DIGIT_BIT) | DIGIT(a, i); + } + #endif + return res; +} +#endif + + + +#ifdef BN_MP_GROW_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* grow as required */ +int mp_grow(mp_int *a, int size) { + int i; + mp_digit *tmp; + + /* if the alloc size is smaller alloc more ram */ + if (a->alloc < size) { + /* ensure there are always at least MP_PREC digits extra on top */ + size += (MP_PREC * 2) - (size % MP_PREC); + + /* reallocate the array a->dp + * + * We store the return in a temporary variable + * in case the operation failed we don't want + * to overwrite the dp member of a. + */ + tmp = OPT_CAST(mp_digit) XREALLOC(a->dp, sizeof(mp_digit) * size); + if (tmp == NULL) { + /* reallocation failed but "a" is still valid [can be freed] */ + return MP_MEM; + } + + /* reallocation succeeded so set a->dp */ + a->dp = tmp; + + /* zero excess digits */ + i = a->alloc; + a->alloc = size; + for ( ; i < a->alloc; i++) { + a->dp[i] = 0; + } + } + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_IMPORT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* based on gmp's mpz_import. + * see http://gmplib.org/manual/Integer-Import-and-Export.html + */ +int mp_import(mp_int *rop, size_t count, int order, size_t size, + int endian, size_t nails, const void *op) { + int result; + size_t odd_nails, nail_bytes, i, j; + unsigned char odd_nail_mask; + + mp_zero(rop); + + if (endian == 0) { + union { + unsigned int i; + char c[4]; + } lint; + lint.i = 0x01020304; + + endian = (lint.c[0] == 4) ? -1 : 1; + } + + odd_nails = (nails % 8); + odd_nail_mask = 0xff; + for (i = 0; i < odd_nails; ++i) { + odd_nail_mask ^= (1 << (7 - i)); + } + nail_bytes = nails / 8; + + for (i = 0; i < count; ++i) { + for (j = 0; j < (size - nail_bytes); ++j) { + unsigned char byte = *( + (unsigned char *)op + + (((order == 1) ? i : ((count - 1) - i)) * size) + + ((endian == 1) ? (j + nail_bytes) : (((size - 1) - j) - nail_bytes)) + ); + + if ( + (result = mp_mul_2d(rop, ((j == 0) ? (8 - odd_nails) : 8), rop)) != MP_OKAY) { + return result; + } + + rop->dp[0] |= (j == 0) ? (byte & odd_nail_mask) : byte; + rop->used += 1; + } + } + + mp_clamp(rop); + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_INIT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* init a new mp_int */ +int mp_init(mp_int *a) { + int i; + + /* allocate memory required and clear it */ + a->dp = OPT_CAST(mp_digit) XMALLOC(sizeof(mp_digit) * MP_PREC); + if (a->dp == NULL) { + return MP_MEM; + } + + /* set the digits to zero */ + for (i = 0; i < MP_PREC; i++) { + a->dp[i] = 0; + } + + /* set the used to zero, allocated digits to the default precision + * and sign to positive */ + a->used = 0; + a->alloc = MP_PREC; + a->sign = MP_ZPOS; + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_INIT_COPY_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* creates "a" then copies b into it */ +int mp_init_copy(mp_int *a, mp_int *b) { + int res; + + if ((res = mp_init_size(a, b->used)) != MP_OKAY) { + return res; + } + return mp_copy(b, a); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_INIT_MULTI_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ +#include + +int mp_init_multi(mp_int *mp, ...) { + mp_err res = MP_OKAY; /* Assume ok until proven otherwise */ + int n = 0; /* Number of ok inits */ + mp_int *cur_arg = mp; + va_list args; + + va_start(args, mp); /* init args to next argument from caller */ + while (cur_arg != NULL) { + if (mp_init(cur_arg) != MP_OKAY) { + /* Oops - error! Back-track and mp_clear what we already + succeeded in init-ing, then return error. + */ + va_list clean_args; + + /* end the current list */ + va_end(args); + + /* now start cleaning up */ + cur_arg = mp; + va_start(clean_args, mp); + while (n-- != 0) { + mp_clear(cur_arg); + cur_arg = va_arg(clean_args, mp_int *); + } + va_end(clean_args); + res = MP_MEM; + break; + } + n++; + cur_arg = va_arg(args, mp_int *); + } + va_end(args); + return res; /* Assumed ok, if error flagged above. */ +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_INIT_SET_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* initialize and set a digit */ +int mp_init_set(mp_int *a, mp_digit b) { + int err; + + if ((err = mp_init(a)) != MP_OKAY) { + return err; + } + mp_set(a, b); + return err; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_INIT_SET_INT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* initialize and set a digit */ +int mp_init_set_int(mp_int *a, unsigned long b) { + int err; + + if ((err = mp_init(a)) != MP_OKAY) { + return err; + } + return mp_set_int(a, b); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_INIT_SIZE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* init an mp_init for a given size */ +int mp_init_size(mp_int *a, int size) { + int x; + + /* pad size so there are always extra digits */ + size += (MP_PREC * 2) - (size % MP_PREC); + + /* alloc mem */ + a->dp = OPT_CAST(mp_digit) XMALLOC(sizeof(mp_digit) * size); + if (a->dp == NULL) { + return MP_MEM; + } + + /* set the members */ + a->used = 0; + a->alloc = size; + a->sign = MP_ZPOS; + + /* zero the digits */ + for (x = 0; x < size; x++) { + a->dp[x] = 0; + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_INVMOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* hac 14.61, pp608 */ +int mp_invmod(mp_int *a, mp_int *b, mp_int *c) { + /* b cannot be negative */ + if ((b->sign == MP_NEG) || (mp_iszero(b) == MP_YES)) { + return MP_VAL; + } + + #ifdef BN_FAST_MP_INVMOD_C + /* if the modulus is odd we can use a faster routine instead */ + if (mp_isodd(b) == MP_YES) { + return fast_mp_invmod(a, b, c); + } + #endif + + #ifdef BN_MP_INVMOD_SLOW_C + return mp_invmod_slow(a, b, c); + #else + return MP_VAL; + #endif +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_INVMOD_SLOW_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* hac 14.61, pp608 */ +int mp_invmod_slow(mp_int *a, mp_int *b, mp_int *c) { + mp_int x, y, u, v, A, B, C, D; + int res; + + /* b cannot be negative */ + if ((b->sign == MP_NEG) || (mp_iszero(b) == MP_YES)) { + return MP_VAL; + } + + /* init temps */ + if ((res = mp_init_multi(&x, &y, &u, &v, + &A, &B, &C, &D, NULL)) != MP_OKAY) { + return res; + } + + /* x = a, y = b */ + if ((res = mp_mod(a, b, &x)) != MP_OKAY) { + goto LBL_ERR; + } + if ((res = mp_copy(b, &y)) != MP_OKAY) { + goto LBL_ERR; + } + + /* 2. [modified] if x,y are both even then return an error! */ + if ((mp_iseven(&x) == MP_YES) && (mp_iseven(&y) == MP_YES)) { + res = MP_VAL; + goto LBL_ERR; + } + + /* 3. u=x, v=y, A=1, B=0, C=0,D=1 */ + if ((res = mp_copy(&x, &u)) != MP_OKAY) { + goto LBL_ERR; + } + if ((res = mp_copy(&y, &v)) != MP_OKAY) { + goto LBL_ERR; + } + mp_set(&A, 1); + mp_set(&D, 1); + +top: + /* 4. while u is even do */ + while (mp_iseven(&u) == MP_YES) { + /* 4.1 u = u/2 */ + if ((res = mp_div_2(&u, &u)) != MP_OKAY) { + goto LBL_ERR; + } + /* 4.2 if A or B is odd then */ + if ((mp_isodd(&A) == MP_YES) || (mp_isodd(&B) == MP_YES)) { + /* A = (A+y)/2, B = (B-x)/2 */ + if ((res = mp_add(&A, &y, &A)) != MP_OKAY) { + goto LBL_ERR; + } + if ((res = mp_sub(&B, &x, &B)) != MP_OKAY) { + goto LBL_ERR; + } + } + /* A = A/2, B = B/2 */ + if ((res = mp_div_2(&A, &A)) != MP_OKAY) { + goto LBL_ERR; + } + if ((res = mp_div_2(&B, &B)) != MP_OKAY) { + goto LBL_ERR; + } + } + + /* 5. while v is even do */ + while (mp_iseven(&v) == MP_YES) { + /* 5.1 v = v/2 */ + if ((res = mp_div_2(&v, &v)) != MP_OKAY) { + goto LBL_ERR; + } + /* 5.2 if C or D is odd then */ + if ((mp_isodd(&C) == MP_YES) || (mp_isodd(&D) == MP_YES)) { + /* C = (C+y)/2, D = (D-x)/2 */ + if ((res = mp_add(&C, &y, &C)) != MP_OKAY) { + goto LBL_ERR; + } + if ((res = mp_sub(&D, &x, &D)) != MP_OKAY) { + goto LBL_ERR; + } + } + /* C = C/2, D = D/2 */ + if ((res = mp_div_2(&C, &C)) != MP_OKAY) { + goto LBL_ERR; + } + if ((res = mp_div_2(&D, &D)) != MP_OKAY) { + goto LBL_ERR; + } + } + + /* 6. if u >= v then */ + if (mp_cmp(&u, &v) != MP_LT) { + /* u = u - v, A = A - C, B = B - D */ + if ((res = mp_sub(&u, &v, &u)) != MP_OKAY) { + goto LBL_ERR; + } + + if ((res = mp_sub(&A, &C, &A)) != MP_OKAY) { + goto LBL_ERR; + } + + if ((res = mp_sub(&B, &D, &B)) != MP_OKAY) { + goto LBL_ERR; + } + } else { + /* v - v - u, C = C - A, D = D - B */ + if ((res = mp_sub(&v, &u, &v)) != MP_OKAY) { + goto LBL_ERR; + } + + if ((res = mp_sub(&C, &A, &C)) != MP_OKAY) { + goto LBL_ERR; + } + + if ((res = mp_sub(&D, &B, &D)) != MP_OKAY) { + goto LBL_ERR; + } + } + + /* if not zero goto step 4 */ + if (mp_iszero(&u) == MP_NO) + goto top; + + /* now a = C, b = D, gcd == g*v */ + + /* if v != 1 then there is no inverse */ + if (mp_cmp_d(&v, 1) != MP_EQ) { + res = MP_VAL; + goto LBL_ERR; + } + + /* if its too low */ + while (mp_cmp_d(&C, 0) == MP_LT) { + if ((res = mp_add(&C, b, &C)) != MP_OKAY) { + goto LBL_ERR; + } + } + + /* too big */ + while (mp_cmp_mag(&C, b) != MP_LT) { + if ((res = mp_sub(&C, b, &C)) != MP_OKAY) { + goto LBL_ERR; + } + } + + /* C is now the inverse */ + mp_exch(&C, c); + res = MP_OKAY; +LBL_ERR: mp_clear_multi(&x, &y, &u, &v, &A, &B, &C, &D, NULL); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_IS_SQUARE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* Check if remainders are possible squares - fast exclude non-squares */ +static const char rem_128[128] = { + 0, 0, 1, 1, 0, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, + 0, 0, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, + 1, 0, 1, 1, 0, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, + 1, 0, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, + 0, 0, 1, 1, 0, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, + 1, 0, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, + 1, 0, 1, 1, 0, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, + 1, 0, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1 +}; + +static const char rem_105[105] = { + 0, 0, 1, 1, 0, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, + 0, 0, 1, 1, 1, 1, 0, 1, 1, 1, 0, 1, 1, 1, 1, + 0, 1, 1, 1, 1, 1, 0, 1, 1, 0, 1, 1, 1, 1, 1, + 1, 0, 1, 1, 0, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, + 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, + 1, 1, 1, 1, 0, 1, 0, 1, 1, 0, 0, 1, 1, 1, 1, + 1, 0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 1, 1 +}; + +/* Store non-zero to ret if arg is square, and zero if not */ +int mp_is_square(mp_int *arg, int *ret) { + int res; + mp_digit c; + mp_int t; + unsigned long r; + + /* Default to Non-square :) */ + *ret = MP_NO; + + if (arg->sign == MP_NEG) { + return MP_VAL; + } + + /* digits used? (TSD) */ + if (arg->used == 0) { + return MP_OKAY; + } + + /* First check mod 128 (suppose that DIGIT_BIT is at least 7) */ + if (rem_128[127 & DIGIT(arg, 0)] == 1) { + return MP_OKAY; + } + + /* Next check mod 105 (3*5*7) */ + if ((res = mp_mod_d(arg, 105, &c)) != MP_OKAY) { + return res; + } + if (rem_105[c] == 1) { + return MP_OKAY; + } + + + if ((res = mp_init_set_int(&t, 11L * 13L * 17L * 19L * 23L * 29L * 31L)) != MP_OKAY) { + return res; + } + if ((res = mp_mod(arg, &t, &t)) != MP_OKAY) { + goto ERR; + } + r = mp_get_int(&t); + + /* Check for other prime modules, note it's not an ERROR but we must + * free "t" so the easiest way is to goto ERR. We know that res + * is already equal to MP_OKAY from the mp_mod call + */ + if (((1L << (r % 11)) & 0x5C4L) != 0L) goto ERR; + if (((1L << (r % 13)) & 0x9E4L) != 0L) goto ERR; + if (((1L << (r % 17)) & 0x5CE8L) != 0L) goto ERR; + if (((1L << (r % 19)) & 0x4F50CL) != 0L) goto ERR; + if (((1L << (r % 23)) & 0x7ACCA0L) != 0L) goto ERR; + if (((1L << (r % 29)) & 0xC2EDD0CL) != 0L) goto ERR; + if (((1L << (r % 31)) & 0x6DE2B848L) != 0L) goto ERR; + + /* Final check - is sqr(sqrt(arg)) == arg ? */ + if ((res = mp_sqrt(arg, &t)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sqr(&t, &t)) != MP_OKAY) { + goto ERR; + } + + *ret = (mp_cmp_mag(&t, arg) == MP_EQ) ? MP_YES : MP_NO; +ERR: mp_clear(&t); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_JACOBI_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* computes the jacobi c = (a | n) (or Legendre if n is prime) + * HAC pp. 73 Algorithm 2.149 + * HAC is wrong here, as the special case of (0 | 1) is not + * handled correctly. + */ +int mp_jacobi(mp_int *a, mp_int *n, int *c) { + mp_int a1, p1; + int k, s, r, res; + mp_digit residue; + + /* if n <= 0 return MP_VAL */ + if (mp_cmp_d(n, 0) != MP_GT) { + return MP_VAL; + } + + /* step 1. handle case of a == 0 */ + if (mp_iszero(a) == MP_YES) { + /* special case of a == 0 and n == 1 */ + if (mp_cmp_d(n, 1) == MP_EQ) { + *c = 1; + } else { + *c = 0; + } + return MP_OKAY; + } + + /* step 2. if a == 1, return 1 */ + if (mp_cmp_d(a, 1) == MP_EQ) { + *c = 1; + return MP_OKAY; + } + + /* default */ + s = 0; + + /* step 3. write a = a1 * 2**k */ + if ((res = mp_init_copy(&a1, a)) != MP_OKAY) { + return res; + } + + if ((res = mp_init(&p1)) != MP_OKAY) { + goto LBL_A1; + } + + /* divide out larger power of two */ + k = mp_cnt_lsb(&a1); + if ((res = mp_div_2d(&a1, k, &a1, NULL)) != MP_OKAY) { + goto LBL_P1; + } + + /* step 4. if e is even set s=1 */ + if ((k & 1) == 0) { + s = 1; + } else { + /* else set s=1 if p = 1/7 (mod 8) or s=-1 if p = 3/5 (mod 8) */ + residue = n->dp[0] & 7; + + if ((residue == 1) || (residue == 7)) { + s = 1; + } else if ((residue == 3) || (residue == 5)) { + s = -1; + } + } + + /* step 5. if p == 3 (mod 4) *and* a1 == 3 (mod 4) then s = -s */ + if (((n->dp[0] & 3) == 3) && ((a1.dp[0] & 3) == 3)) { + s = -s; + } + + /* if a1 == 1 we're done */ + if (mp_cmp_d(&a1, 1) == MP_EQ) { + *c = s; + } else { + /* n1 = n mod a1 */ + if ((res = mp_mod(n, &a1, &p1)) != MP_OKAY) { + goto LBL_P1; + } + if ((res = mp_jacobi(&p1, &a1, &r)) != MP_OKAY) { + goto LBL_P1; + } + *c = s * r; + } + + /* done */ + res = MP_OKAY; +LBL_P1: mp_clear(&p1); +LBL_A1: mp_clear(&a1); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_KARATSUBA_MUL_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* c = |a| * |b| using Karatsuba Multiplication using + * three half size multiplications + * + * Let B represent the radix [e.g. 2**DIGIT_BIT] and + * let n represent half of the number of digits in + * the min(a,b) + * + * a = a1 * B**n + a0 + * b = b1 * B**n + b0 + * + * Then, a * b => + a1b1 * B**2n + ((a1 + a0)(b1 + b0) - (a0b0 + a1b1)) * B + a0b0 + * + * Note that a1b1 and a0b0 are used twice and only need to be + * computed once. So in total three half size (half # of + * digit) multiplications are performed, a0b0, a1b1 and + * (a1+b1)(a0+b0) + * + * Note that a multiplication of half the digits requires + * 1/4th the number of single precision multiplications so in + * total after one call 25% of the single precision multiplications + * are saved. Note also that the call to mp_mul can end up back + * in this function if the a0, a1, b0, or b1 are above the threshold. + * This is known as divide-and-conquer and leads to the famous + * O(N**lg(3)) or O(N**1.584) work which is asymptopically lower than + * the standard O(N**2) that the baseline/comba methods use. + * Generally though the overhead of this method doesn't pay off + * until a certain size (N ~ 80) is reached. + */ +int mp_karatsuba_mul(mp_int *a, mp_int *b, mp_int *c) { + mp_int x0, x1, y0, y1, t1, x0y0, x1y1; + int B, err; + + /* default the return code to an error */ + err = MP_MEM; + + /* min # of digits */ + B = MIN(a->used, b->used); + + /* now divide in two */ + B = B >> 1; + + /* init copy all the temps */ + if (mp_init_size(&x0, B) != MP_OKAY) + goto ERR; + if (mp_init_size(&x1, a->used - B) != MP_OKAY) + goto X0; + if (mp_init_size(&y0, B) != MP_OKAY) + goto X1; + if (mp_init_size(&y1, b->used - B) != MP_OKAY) + goto Y0; + + /* init temps */ + if (mp_init_size(&t1, B * 2) != MP_OKAY) + goto Y1; + if (mp_init_size(&x0y0, B * 2) != MP_OKAY) + goto T1; + if (mp_init_size(&x1y1, B * 2) != MP_OKAY) + goto X0Y0; + + /* now shift the digits */ + x0.used = y0.used = B; + x1.used = a->used - B; + y1.used = b->used - B; + + { + int x; + mp_digit *tmpa, *tmpb, *tmpx, *tmpy; + + /* we copy the digits directly instead of using higher level functions + * since we also need to shift the digits + */ + tmpa = a->dp; + tmpb = b->dp; + + tmpx = x0.dp; + tmpy = y0.dp; + for (x = 0; x < B; x++) { + *tmpx++ = *tmpa++; + *tmpy++ = *tmpb++; + } + + tmpx = x1.dp; + for (x = B; x < a->used; x++) { + *tmpx++ = *tmpa++; + } + + tmpy = y1.dp; + for (x = B; x < b->used; x++) { + *tmpy++ = *tmpb++; + } + } + + /* only need to clamp the lower words since by definition the + * upper words x1/y1 must have a known number of digits + */ + mp_clamp(&x0); + mp_clamp(&y0); + + /* now calc the products x0y0 and x1y1 */ + /* after this x0 is no longer required, free temp [x0==t2]! */ + if (mp_mul(&x0, &y0, &x0y0) != MP_OKAY) + goto X1Y1; /* x0y0 = x0*y0 */ + if (mp_mul(&x1, &y1, &x1y1) != MP_OKAY) + goto X1Y1; /* x1y1 = x1*y1 */ + + /* now calc x1+x0 and y1+y0 */ + if (s_mp_add(&x1, &x0, &t1) != MP_OKAY) + goto X1Y1; /* t1 = x1 - x0 */ + if (s_mp_add(&y1, &y0, &x0) != MP_OKAY) + goto X1Y1; /* t2 = y1 - y0 */ + if (mp_mul(&t1, &x0, &t1) != MP_OKAY) + goto X1Y1; /* t1 = (x1 + x0) * (y1 + y0) */ + + /* add x0y0 */ + if (mp_add(&x0y0, &x1y1, &x0) != MP_OKAY) + goto X1Y1; /* t2 = x0y0 + x1y1 */ + if (s_mp_sub(&t1, &x0, &t1) != MP_OKAY) + goto X1Y1; /* t1 = (x1+x0)*(y1+y0) - (x1y1 + x0y0) */ + + /* shift by B */ + if (mp_lshd(&t1, B) != MP_OKAY) + goto X1Y1; /* t1 = (x0y0 + x1y1 - (x1-x0)*(y1-y0))<used; + + /* now divide in two */ + B = B >> 1; + + /* init copy all the temps */ + if (mp_init_size(&x0, B) != MP_OKAY) + goto ERR; + if (mp_init_size(&x1, a->used - B) != MP_OKAY) + goto X0; + + /* init temps */ + if (mp_init_size(&t1, a->used * 2) != MP_OKAY) + goto X1; + if (mp_init_size(&t2, a->used * 2) != MP_OKAY) + goto T1; + if (mp_init_size(&x0x0, B * 2) != MP_OKAY) + goto T2; + if (mp_init_size(&x1x1, (a->used - B) * 2) != MP_OKAY) + goto X0X0; + + { + int x; + mp_digit *dst, *src; + + src = a->dp; + + /* now shift the digits */ + dst = x0.dp; + for (x = 0; x < B; x++) { + *dst++ = *src++; + } + + dst = x1.dp; + for (x = B; x < a->used; x++) { + *dst++ = *src++; + } + } + + x0.used = B; + x1.used = a->used - B; + + mp_clamp(&x0); + + /* now calc the products x0*x0 and x1*x1 */ + if (mp_sqr(&x0, &x0x0) != MP_OKAY) + goto X1X1; /* x0x0 = x0*x0 */ + if (mp_sqr(&x1, &x1x1) != MP_OKAY) + goto X1X1; /* x1x1 = x1*x1 */ + + /* now calc (x1+x0)**2 */ + if (s_mp_add(&x1, &x0, &t1) != MP_OKAY) + goto X1X1; /* t1 = x1 - x0 */ + if (mp_sqr(&t1, &t1) != MP_OKAY) + goto X1X1; /* t1 = (x1 - x0) * (x1 - x0) */ + + /* add x0y0 */ + if (s_mp_add(&x0x0, &x1x1, &t2) != MP_OKAY) + goto X1X1; /* t2 = x0x0 + x1x1 */ + if (s_mp_sub(&t1, &t2, &t1) != MP_OKAY) + goto X1X1; /* t1 = (x1+x0)**2 - (x0x0 + x1x1) */ + + /* shift by B */ + if (mp_lshd(&t1, B) != MP_OKAY) + goto X1X1; /* t1 = (x0x0 + x1x1 - (x1-x0)*(x1-x0))<sign = MP_ZPOS; + +LBL_T: + mp_clear_multi(&t1, &t2, NULL); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_LSHD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* shift left a certain amount of digits */ +int mp_lshd(mp_int *a, int b) { + int x, res; + + /* if its less than zero return */ + if (b <= 0) { + return MP_OKAY; + } + + /* grow to fit the new digits */ + if (a->alloc < (a->used + b)) { + if ((res = mp_grow(a, a->used + b)) != MP_OKAY) { + return res; + } + } + + { + mp_digit *top, *bottom; + + /* increment the used by the shift amount then copy upwards */ + a->used += b; + + /* top */ + top = a->dp + a->used - 1; + + /* base */ + bottom = (a->dp + a->used - 1) - b; + + /* much like mp_rshd this is implemented using a sliding window + * except the window goes the otherway around. Copying from + * the bottom to the top. see bn_mp_rshd.c for more info. + */ + for (x = a->used - 1; x >= b; x--) { + *top-- = *bottom--; + } + + /* zero the lower digits */ + top = a->dp; + for (x = 0; x < b; x++) { + *top++ = 0; + } + } + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* c = a mod b, 0 <= c < b if b > 0, b < c <= 0 if b < 0 */ +int +mp_mod(mp_int *a, mp_int *b, mp_int *c) { + mp_int t; + int res; + + if ((res = mp_init(&t)) != MP_OKAY) { + return res; + } + + if ((res = mp_div(a, b, NULL, &t)) != MP_OKAY) { + mp_clear(&t); + return res; + } + + if ((mp_iszero(&t) != MP_NO) || (t.sign == b->sign)) { + res = MP_OKAY; + mp_exch(&t, c); + } else { + res = mp_add(b, &t, c); + } + + mp_clear(&t); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MOD_2D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* calc a value mod 2**b */ +int +mp_mod_2d(mp_int *a, int b, mp_int *c) { + int x, res; + + /* if b is <= 0 then zero the int */ + if (b <= 0) { + mp_zero(c); + return MP_OKAY; + } + + /* if the modulus is larger than the value than return */ + if (b >= (int)(a->used * DIGIT_BIT)) { + res = mp_copy(a, c); + return res; + } + + /* copy */ + if ((res = mp_copy(a, c)) != MP_OKAY) { + return res; + } + + /* zero digits above the last digit of the modulus */ + for (x = (b / DIGIT_BIT) + (((b % DIGIT_BIT) == 0) ? 0 : 1); x < c->used; x++) { + c->dp[x] = 0; + } + /* clear the digit that is not completely outside/inside the modulus */ + c->dp[b / DIGIT_BIT] &= + (mp_digit)((((mp_digit)1) << (((mp_digit)b) % DIGIT_BIT)) - ((mp_digit)1)); + mp_clamp(c); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MOD_D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +int +mp_mod_d(mp_int *a, mp_digit b, mp_digit *c) { + return mp_div_d(a, b, NULL, c); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MONTGOMERY_CALC_NORMALIZATION_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* + * shifts with subtractions when the result is greater than b. + * + * The method is slightly modified to shift B unconditionally upto just under + * the leading bit of b. This saves alot of multiple precision shifting. + */ +int mp_montgomery_calc_normalization(mp_int *a, mp_int *b) { + int x, bits, res; + + /* how many bits of last digit does b use */ + bits = mp_count_bits(b) % DIGIT_BIT; + + if (b->used > 1) { + if ((res = mp_2expt(a, ((b->used - 1) * DIGIT_BIT) + bits - 1)) != MP_OKAY) { + return res; + } + } else { + mp_set(a, 1); + bits = 1; + } + + + /* now compute C = A * B mod b */ + for (x = bits - 1; x < (int)DIGIT_BIT; x++) { + if ((res = mp_mul_2(a, a)) != MP_OKAY) { + return res; + } + if (mp_cmp_mag(a, b) != MP_LT) { + if ((res = s_mp_sub(a, b, a)) != MP_OKAY) { + return res; + } + } + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MONTGOMERY_REDUCE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* computes xR**-1 == x (mod N) via Montgomery Reduction */ +int +mp_montgomery_reduce(mp_int *x, mp_int *n, mp_digit rho) { + int ix, res, digs; + mp_digit mu; + + /* can the fast reduction [comba] method be used? + * + * Note that unlike in mul you're safely allowed *less* + * than the available columns [255 per default] since carries + * are fixed up in the inner loop. + */ + digs = (n->used * 2) + 1; + if ((digs < MP_WARRAY) && + (n->used < + (1 << ((CHAR_BIT * sizeof(mp_word)) - (2 * DIGIT_BIT))))) { + return fast_mp_montgomery_reduce(x, n, rho); + } + + /* grow the input as required */ + if (x->alloc < digs) { + if ((res = mp_grow(x, digs)) != MP_OKAY) { + return res; + } + } + x->used = digs; + + for (ix = 0; ix < n->used; ix++) { + /* mu = ai * rho mod b + * + * The value of rho must be precalculated via + * montgomery_setup() such that + * it equals -1/n0 mod b this allows the + * following inner loop to reduce the + * input one digit at a time + */ + mu = (mp_digit)(((mp_word)x->dp[ix] * (mp_word)rho) & MP_MASK); + + /* a = a + mu * m * b**i */ + { + int iy; + mp_digit *tmpn, *tmpx, u; + mp_word r; + + /* alias for digits of the modulus */ + tmpn = n->dp; + + /* alias for the digits of x [the input] */ + tmpx = x->dp + ix; + + /* set the carry to zero */ + u = 0; + + /* Multiply and add in place */ + for (iy = 0; iy < n->used; iy++) { + /* compute product and sum */ + r = ((mp_word)mu * (mp_word) * tmpn++) + + (mp_word)u + (mp_word) * tmpx; + + /* get carry */ + u = (mp_digit)(r >> ((mp_word)DIGIT_BIT)); + + /* fix digit */ + *tmpx++ = (mp_digit)(r & ((mp_word)MP_MASK)); + } + /* At this point the ix'th digit of x should be zero */ + + + /* propagate carries upwards as required*/ + while (u != 0) { + *tmpx += u; + u = *tmpx >> DIGIT_BIT; + *tmpx++ &= MP_MASK; + } + } + } + + /* at this point the n.used'th least + * significant digits of x are all zero + * which means we can shift x to the + * right by n.used digits and the + * residue is unchanged. + */ + + /* x = x/b**n.used */ + mp_clamp(x); + mp_rshd(x, n->used); + + /* if x >= n then x = x - n */ + if (mp_cmp_mag(x, n) != MP_LT) { + return s_mp_sub(x, n, x); + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MONTGOMERY_SETUP_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* setups the montgomery reduction stuff */ +int +mp_montgomery_setup(mp_int *n, mp_digit *rho) { + mp_digit x, b; + +/* fast inversion mod 2**k + * + * Based on the fact that + * + * XA = 1 (mod 2**n) => (X(2-XA)) A = 1 (mod 2**2n) + * => 2*X*A - X*X*A*A = 1 + * => 2*(1) - (1) = 1 + */ + b = n->dp[0]; + + if ((b & 1) == 0) { + return MP_VAL; + } + + x = (((b + 2) & 4) << 1) + b; /* here x*a==1 mod 2**4 */ + x *= 2 - (b * x); /* here x*a==1 mod 2**8 */ + #if !defined(MP_8BIT) + x *= 2 - (b * x); /* here x*a==1 mod 2**16 */ + #endif + #if defined(MP_64BIT) || !(defined(MP_8BIT) || defined(MP_16BIT)) + x *= 2 - (b * x); /* here x*a==1 mod 2**32 */ + #endif + #ifdef MP_64BIT + x *= 2 - (b * x); /* here x*a==1 mod 2**64 */ + #endif + + /* rho = -1/m mod b */ + *rho = (mp_digit)(((mp_word)1 << ((mp_word)DIGIT_BIT)) - x) & MP_MASK; + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MUL_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* high level multiplication (handles sign) */ +int mp_mul(mp_int *a, mp_int *b, mp_int *c) { + int res, neg; + + neg = (a->sign == b->sign) ? MP_ZPOS : MP_NEG; + + /* use Toom-Cook? */ + #ifdef BN_MP_TOOM_MUL_C + if (MIN(a->used, b->used) >= TOOM_MUL_CUTOFF) { + res = mp_toom_mul(a, b, c); + } else + #endif + #ifdef BN_MP_KARATSUBA_MUL_C + /* use Karatsuba? */ + if (MIN(a->used, b->used) >= KARATSUBA_MUL_CUTOFF) { + res = mp_karatsuba_mul(a, b, c); + } else + #endif + { + /* can we use the fast multiplier? + * + * The fast multiplier can be used if the output will + * have less than MP_WARRAY digits and the number of + * digits won't affect carry propagation + */ + int digs = a->used + b->used + 1; + + #ifdef BN_FAST_S_MP_MUL_DIGS_C + if ((digs < MP_WARRAY) && + (MIN(a->used, b->used) <= + (1 << ((CHAR_BIT * sizeof(mp_word)) - (2 * DIGIT_BIT))))) { + res = fast_s_mp_mul_digs(a, b, c, digs); + } else + #endif + { + #ifdef BN_S_MP_MUL_DIGS_C + res = s_mp_mul(a, b, c); /* uses s_mp_mul_digs */ + #else + res = MP_VAL; + #endif + } + } + c->sign = (c->used > 0) ? neg : MP_ZPOS; + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MUL_2_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* b = a*2 */ +int mp_mul_2(mp_int *a, mp_int *b) { + int x, res, oldused; + + /* grow to accomodate result */ + if (b->alloc < (a->used + 1)) { + if ((res = mp_grow(b, a->used + 1)) != MP_OKAY) { + return res; + } + } + + oldused = b->used; + b->used = a->used; + + { + mp_digit r, rr, *tmpa, *tmpb; + + /* alias for source */ + tmpa = a->dp; + + /* alias for dest */ + tmpb = b->dp; + + /* carry */ + r = 0; + for (x = 0; x < a->used; x++) { + /* get what will be the *next* carry bit from the + * MSB of the current digit + */ + rr = *tmpa >> ((mp_digit)(DIGIT_BIT - 1)); + + /* now shift up this digit, add in the carry [from the previous] */ + *tmpb++ = ((*tmpa++ << ((mp_digit)1)) | r) & MP_MASK; + + /* copy the carry that would be from the source + * digit into the next iteration + */ + r = rr; + } + + /* new leading digit? */ + if (r != 0) { + /* add a MSB which is always 1 at this point */ + *tmpb = 1; + ++(b->used); + } + + /* now zero any excess digits on the destination + * that we didn't write to + */ + tmpb = b->dp + b->used; + for (x = b->used; x < oldused; x++) { + *tmpb++ = 0; + } + } + b->sign = a->sign; + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MUL_2D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* shift left by a certain bit count */ +int mp_mul_2d(mp_int *a, int b, mp_int *c) { + mp_digit d; + int res; + + /* copy */ + if (a != c) { + if ((res = mp_copy(a, c)) != MP_OKAY) { + return res; + } + } + + if (c->alloc < (int)(c->used + (b / DIGIT_BIT) + 1)) { + if ((res = mp_grow(c, c->used + (b / DIGIT_BIT) + 1)) != MP_OKAY) { + return res; + } + } + + /* shift by as many digits in the bit count */ + if (b >= (int)DIGIT_BIT) { + if ((res = mp_lshd(c, b / DIGIT_BIT)) != MP_OKAY) { + return res; + } + } + + /* shift any bit count < DIGIT_BIT */ + d = (mp_digit)(b % DIGIT_BIT); + if (d != 0) { + mp_digit *tmpc, shift, mask, r, rr; + int x; + + /* bitmask for carries */ + mask = (((mp_digit)1) << d) - 1; + + /* shift for msbs */ + shift = DIGIT_BIT - d; + + /* alias */ + tmpc = c->dp; + + /* carry */ + r = 0; + for (x = 0; x < c->used; x++) { + /* get the higher bits of the current word */ + rr = (*tmpc >> shift) & mask; + + /* shift the current word and OR in the carry */ + *tmpc = ((*tmpc << d) | r) & MP_MASK; + ++tmpc; + + /* set the carry to the carry bits of the current word */ + r = rr; + } + + /* set final carry */ + if (r != 0) { + c->dp[(c->used)++] = r; + } + } + mp_clamp(c); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MUL_D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* multiply by a digit */ +int +mp_mul_d(mp_int *a, mp_digit b, mp_int *c) { + mp_digit u, *tmpa, *tmpc; + mp_word r; + int ix, res, olduse; + + /* make sure c is big enough to hold a*b */ + if (c->alloc < (a->used + 1)) { + if ((res = mp_grow(c, a->used + 1)) != MP_OKAY) { + return res; + } + } + + /* get the original destinations used count */ + olduse = c->used; + + /* set the sign */ + c->sign = a->sign; + + /* alias for a->dp [source] */ + tmpa = a->dp; + + /* alias for c->dp [dest] */ + tmpc = c->dp; + + /* zero carry */ + u = 0; + + /* compute columns */ + for (ix = 0; ix < a->used; ix++) { + /* compute product and carry sum for this term */ + r = (mp_word)u + ((mp_word) * tmpa++ *(mp_word)b); + + /* mask off higher bits to get a single digit */ + *tmpc++ = (mp_digit)(r & ((mp_word)MP_MASK)); + + /* send carry into next iteration */ + u = (mp_digit)(r >> ((mp_word)DIGIT_BIT)); + } + + /* store final carry [if any] and increment ix offset */ + *tmpc++ = u; + ++ix; + + /* now zero digits above the top */ + while (ix++ < olduse) { + *tmpc++ = 0; + } + + /* set used count */ + c->used = a->used + 1; + mp_clamp(c); + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_MULMOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* d = a * b (mod c) */ +int mp_mulmod(mp_int *a, mp_int *b, mp_int *c, mp_int *d) { + int res; + mp_int t; + + if ((res = mp_init(&t)) != MP_OKAY) { + return res; + } + + if ((res = mp_mul(a, b, &t)) != MP_OKAY) { + mp_clear(&t); + return res; + } + res = mp_mod(&t, c, d); + mp_clear(&t); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_N_ROOT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* wrapper function for mp_n_root_ex() + * computes c = (a)**(1/b) such that (c)**b <= a and (c+1)**b > a + */ +int mp_n_root(mp_int *a, mp_digit b, mp_int *c) { + return mp_n_root_ex(a, b, c, 0); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_N_ROOT_EX_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* find the n'th root of an integer + * + * Result found such that (c)**b <= a and (c+1)**b > a + * + * This algorithm uses Newton's approximation + * x[i+1] = x[i] - f(x[i])/f'(x[i]) + * which will find the root in log(N) time where + * each step involves a fair bit. This is not meant to + * find huge roots [square and cube, etc]. + */ +int mp_n_root_ex(mp_int *a, mp_digit b, mp_int *c, int fast) { + mp_int t1, t2, t3; + int res, neg; + + /* input must be positive if b is even */ + if (((b & 1) == 0) && (a->sign == MP_NEG)) { + return MP_VAL; + } + + if ((res = mp_init(&t1)) != MP_OKAY) { + return res; + } + + if ((res = mp_init(&t2)) != MP_OKAY) { + goto LBL_T1; + } + + if ((res = mp_init(&t3)) != MP_OKAY) { + goto LBL_T2; + } + + /* if a is negative fudge the sign but keep track */ + neg = a->sign; + a->sign = MP_ZPOS; + + /* t2 = 2 */ + mp_set(&t2, 2); + + do { + /* t1 = t2 */ + if ((res = mp_copy(&t2, &t1)) != MP_OKAY) { + goto LBL_T3; + } + + /* t2 = t1 - ((t1**b - a) / (b * t1**(b-1))) */ + + /* t3 = t1**(b-1) */ + if ((res = mp_expt_d_ex(&t1, b - 1, &t3, fast)) != MP_OKAY) { + goto LBL_T3; + } + + /* numerator */ + /* t2 = t1**b */ + if ((res = mp_mul(&t3, &t1, &t2)) != MP_OKAY) { + goto LBL_T3; + } + + /* t2 = t1**b - a */ + if ((res = mp_sub(&t2, a, &t2)) != MP_OKAY) { + goto LBL_T3; + } + + /* denominator */ + /* t3 = t1**(b-1) * b */ + if ((res = mp_mul_d(&t3, b, &t3)) != MP_OKAY) { + goto LBL_T3; + } + + /* t3 = (t1**b - a)/(b * t1**(b-1)) */ + if ((res = mp_div(&t2, &t3, &t3, NULL)) != MP_OKAY) { + goto LBL_T3; + } + + if ((res = mp_sub(&t1, &t3, &t2)) != MP_OKAY) { + goto LBL_T3; + } + } while (mp_cmp(&t1, &t2) != MP_EQ); + + /* result can be off by a few so check */ + for ( ; ; ) { + if ((res = mp_expt_d_ex(&t1, b, &t2, fast)) != MP_OKAY) { + goto LBL_T3; + } + + if (mp_cmp(&t2, a) == MP_GT) { + if ((res = mp_sub_d(&t1, 1, &t1)) != MP_OKAY) { + goto LBL_T3; + } + } else { + break; + } + } + + /* reset the sign of a first */ + a->sign = neg; + + /* set the result */ + mp_exch(&t1, c); + + /* set the sign of the result */ + c->sign = neg; + + res = MP_OKAY; + +LBL_T3: mp_clear(&t3); +LBL_T2: mp_clear(&t2); +LBL_T1: mp_clear(&t1); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_NEG_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* b = -a */ +int mp_neg(mp_int *a, mp_int *b) { + int res; + + if (a != b) { + if ((res = mp_copy(a, b)) != MP_OKAY) { + return res; + } + } + + if (mp_iszero(b) != MP_YES) { + b->sign = (a->sign == MP_ZPOS) ? MP_NEG : MP_ZPOS; + } else { + b->sign = MP_ZPOS; + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_OR_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* OR two ints together */ +int mp_or(mp_int *a, mp_int *b, mp_int *c) { + int res, ix, px; + mp_int t, *x; + + if (a->used > b->used) { + if ((res = mp_init_copy(&t, a)) != MP_OKAY) { + return res; + } + px = b->used; + x = b; + } else { + if ((res = mp_init_copy(&t, b)) != MP_OKAY) { + return res; + } + px = a->used; + x = a; + } + + for (ix = 0; ix < px; ix++) { + t.dp[ix] |= x->dp[ix]; + } + mp_clamp(&t); + mp_exch(c, &t); + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_PRIME_FERMAT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* performs one Fermat test. + * + * If "a" were prime then b**a == b (mod a) since the order of + * the multiplicative sub-group would be phi(a) = a-1. That means + * it would be the same as b**(a mod (a-1)) == b**1 == b (mod a). + * + * Sets result to 1 if the congruence holds, or zero otherwise. + */ +int mp_prime_fermat(mp_int *a, mp_int *b, int *result) { + mp_int t; + int err; + + /* default to composite */ + *result = MP_NO; + + /* ensure b > 1 */ + if (mp_cmp_d(b, 1) != MP_GT) { + return MP_VAL; + } + + /* init t */ + if ((err = mp_init(&t)) != MP_OKAY) { + return err; + } + + /* compute t = b**a mod a */ + if ((err = mp_exptmod(b, a, a, &t)) != MP_OKAY) { + goto LBL_T; + } + + /* is it equal to b? */ + if (mp_cmp(&t, b) == MP_EQ) { + *result = MP_YES; + } + + err = MP_OKAY; +LBL_T: mp_clear(&t); + return err; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_PRIME_IS_DIVISIBLE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* determines if an integers is divisible by one + * of the first PRIME_SIZE primes or not + * + * sets result to 0 if not, 1 if yes + */ +int mp_prime_is_divisible(mp_int *a, int *result) { + int err, ix; + mp_digit res; + + /* default to not */ + *result = MP_NO; + + for (ix = 0; ix < PRIME_SIZE; ix++) { + /* what is a mod LBL_prime_tab[ix] */ + if ((err = mp_mod_d(a, ltm_prime_tab[ix], &res)) != MP_OKAY) { + return err; + } + + /* is the residue zero? */ + if (res == 0) { + *result = MP_YES; + return MP_OKAY; + } + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_PRIME_IS_PRIME_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* performs a variable number of rounds of Miller-Rabin + * + * Probability of error after t rounds is no more than + + * + * Sets result to 1 if probably prime, 0 otherwise + */ +int mp_prime_is_prime(mp_int *a, int t, int *result) { + mp_int b; + int ix, err, res; + + /* default to no */ + *result = MP_NO; + + /* valid value of t? */ + if ((t <= 0) || (t > PRIME_SIZE)) { + return MP_VAL; + } + + /* is the input equal to one of the primes in the table? */ + for (ix = 0; ix < PRIME_SIZE; ix++) { + if (mp_cmp_d(a, ltm_prime_tab[ix]) == MP_EQ) { + *result = 1; + return MP_OKAY; + } + } + + /* first perform trial division */ + if ((err = mp_prime_is_divisible(a, &res)) != MP_OKAY) { + return err; + } + + /* return if it was trivially divisible */ + if (res == MP_YES) { + return MP_OKAY; + } + + /* now perform the miller-rabin rounds */ + if ((err = mp_init(&b)) != MP_OKAY) { + return err; + } + + for (ix = 0; ix < t; ix++) { + /* set the prime */ + mp_set(&b, ltm_prime_tab[ix]); + + if ((err = mp_prime_miller_rabin(a, &b, &res)) != MP_OKAY) { + goto LBL_B; + } + + if (res == MP_NO) { + goto LBL_B; + } + } + + /* passed the test */ + *result = MP_YES; +LBL_B: mp_clear(&b); + return err; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_PRIME_MILLER_RABIN_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* Miller-Rabin test of "a" to the base of "b" as described in + * HAC pp. 139 Algorithm 4.24 + * + * Sets result to 0 if definitely composite or 1 if probably prime. + * Randomly the chance of error is no more than 1/4 and often + * very much lower. + */ +int mp_prime_miller_rabin(mp_int *a, mp_int *b, int *result) { + mp_int n1, y, r; + int s, j, err; + + /* default */ + *result = MP_NO; + + /* ensure b > 1 */ + if (mp_cmp_d(b, 1) != MP_GT) { + return MP_VAL; + } + + /* get n1 = a - 1 */ + if ((err = mp_init_copy(&n1, a)) != MP_OKAY) { + return err; + } + if ((err = mp_sub_d(&n1, 1, &n1)) != MP_OKAY) { + goto LBL_N1; + } + + /* set 2**s * r = n1 */ + if ((err = mp_init_copy(&r, &n1)) != MP_OKAY) { + goto LBL_N1; + } + + /* count the number of least significant bits + * which are zero + */ + s = mp_cnt_lsb(&r); + + /* now divide n - 1 by 2**s */ + if ((err = mp_div_2d(&r, s, &r, NULL)) != MP_OKAY) { + goto LBL_R; + } + + /* compute y = b**r mod a */ + if ((err = mp_init(&y)) != MP_OKAY) { + goto LBL_R; + } + if ((err = mp_exptmod(b, &r, a, &y)) != MP_OKAY) { + goto LBL_Y; + } + + /* if y != 1 and y != n1 do */ + if ((mp_cmp_d(&y, 1) != MP_EQ) && (mp_cmp(&y, &n1) != MP_EQ)) { + j = 1; + /* while j <= s-1 and y != n1 */ + while ((j <= (s - 1)) && (mp_cmp(&y, &n1) != MP_EQ)) { + if ((err = mp_sqrmod(&y, a, &y)) != MP_OKAY) { + goto LBL_Y; + } + + /* if y == 1 then composite */ + if (mp_cmp_d(&y, 1) == MP_EQ) { + goto LBL_Y; + } + + ++j; + } + + /* if y != n1 then composite */ + if (mp_cmp(&y, &n1) != MP_EQ) { + goto LBL_Y; + } + } + + /* probably prime now */ + *result = MP_YES; +LBL_Y: mp_clear(&y); +LBL_R: mp_clear(&r); +LBL_N1: mp_clear(&n1); + return err; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_PRIME_NEXT_PRIME_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* finds the next prime after the number "a" using "t" trials + * of Miller-Rabin. + * + * bbs_style = 1 means the prime must be congruent to 3 mod 4 + */ +int mp_prime_next_prime(mp_int *a, int t, int bbs_style) { + int err, res = MP_NO, x, y; + mp_digit res_tab[PRIME_SIZE], step, kstep; + mp_int b; + + /* ensure t is valid */ + if ((t <= 0) || (t > PRIME_SIZE)) { + return MP_VAL; + } + + /* force positive */ + a->sign = MP_ZPOS; + + /* simple algo if a is less than the largest prime in the table */ + if (mp_cmp_d(a, ltm_prime_tab[PRIME_SIZE - 1]) == MP_LT) { + /* find which prime it is bigger than */ + for (x = PRIME_SIZE - 2; x >= 0; x--) { + if (mp_cmp_d(a, ltm_prime_tab[x]) != MP_LT) { + if (bbs_style == 1) { + /* ok we found a prime smaller or + * equal [so the next is larger] + * + * however, the prime must be + * congruent to 3 mod 4 + */ + if ((ltm_prime_tab[x + 1] & 3) != 3) { + /* scan upwards for a prime congruent to 3 mod 4 */ + for (y = x + 1; y < PRIME_SIZE; y++) { + if ((ltm_prime_tab[y] & 3) == 3) { + mp_set(a, ltm_prime_tab[y]); + return MP_OKAY; + } + } + } + } else { + mp_set(a, ltm_prime_tab[x + 1]); + return MP_OKAY; + } + } + } + /* at this point a maybe 1 */ + if (mp_cmp_d(a, 1) == MP_EQ) { + mp_set(a, 2); + return MP_OKAY; + } + /* fall through to the sieve */ + } + + /* generate a prime congruent to 3 mod 4 or 1/3 mod 4? */ + if (bbs_style == 1) { + kstep = 4; + } else { + kstep = 2; + } + + /* at this point we will use a combination of a sieve and Miller-Rabin */ + + if (bbs_style == 1) { + /* if a mod 4 != 3 subtract the correct value to make it so */ + if ((a->dp[0] & 3) != 3) { + if ((err = mp_sub_d(a, (a->dp[0] & 3) + 1, a)) != MP_OKAY) { + return err; + } + } + } else { + if (mp_iseven(a) == MP_YES) { + /* force odd */ + if ((err = mp_sub_d(a, 1, a)) != MP_OKAY) { + return err; + } + } + } + + /* generate the restable */ + for (x = 1; x < PRIME_SIZE; x++) { + if ((err = mp_mod_d(a, ltm_prime_tab[x], res_tab + x)) != MP_OKAY) { + return err; + } + } + + /* init temp used for Miller-Rabin Testing */ + if ((err = mp_init(&b)) != MP_OKAY) { + return err; + } + + for ( ; ; ) { + /* skip to the next non-trivially divisible candidate */ + step = 0; + do { + /* y == 1 if any residue was zero [e.g. cannot be prime] */ + y = 0; + + /* increase step to next candidate */ + step += kstep; + + /* compute the new residue without using division */ + for (x = 1; x < PRIME_SIZE; x++) { + /* add the step to each residue */ + res_tab[x] += kstep; + + /* subtract the modulus [instead of using division] */ + if (res_tab[x] >= ltm_prime_tab[x]) { + res_tab[x] -= ltm_prime_tab[x]; + } + + /* set flag if zero */ + if (res_tab[x] == 0) { + y = 1; + } + } + } while ((y == 1) && (step < ((((mp_digit)1) << DIGIT_BIT) - kstep))); + + /* add the step */ + if ((err = mp_add_d(a, step, a)) != MP_OKAY) { + goto LBL_ERR; + } + + /* if didn't pass sieve and step == MAX then skip test */ + if ((y == 1) && (step >= ((((mp_digit)1) << DIGIT_BIT) - kstep))) { + continue; + } + + /* is this prime? */ + for (x = 0; x < t; x++) { + mp_set(&b, ltm_prime_tab[x]); + if ((err = mp_prime_miller_rabin(a, &b, &res)) != MP_OKAY) { + goto LBL_ERR; + } + if (res == MP_NO) { + break; + } + } + + if (res == MP_YES) { + break; + } + } + + err = MP_OKAY; +LBL_ERR: + mp_clear(&b); + return err; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_PRIME_RABIN_MILLER_TRIALS_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + + +static const struct { + int k, t; +} sizes[] = { + { 128, 28 }, + { 256, 16 }, + { 384, 10 }, + { 512, 7 }, + { 640, 6 }, + { 768, 5 }, + { 896, 4 }, + { 1024, 4 } +}; + +/* returns # of RM trials required for a given bit size */ +int mp_prime_rabin_miller_trials(int size) { + int x; + + for (x = 0; x < (int)(sizeof(sizes) / (sizeof(sizes[0]))); x++) { + if (sizes[x].k == size) { + return sizes[x].t; + } else if (sizes[x].k > size) { + return (x == 0) ? sizes[0].t : sizes[x - 1].t; + } + } + return sizes[x - 1].t + 1; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_PRIME_RANDOM_EX_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* makes a truly random prime of a given size (bits), + * + * Flags are as follows: + * + * LTM_PRIME_BBS - make prime congruent to 3 mod 4 + * LTM_PRIME_SAFE - make sure (p-1)/2 is prime as well (implies LTM_PRIME_BBS) + * LTM_PRIME_2MSB_ON - make the 2nd highest bit one + * + * You have to supply a callback which fills in a buffer with random bytes. "dat" is a parameter you can + * have passed to the callback (e.g. a state or something). This function doesn't use "dat" itself + * so it can be NULL + * + */ + +/* This is possibly the mother of all prime generation functions, muahahahahaha! */ +int mp_prime_random_ex(mp_int *a, int t, int size, int flags, ltm_prime_callback cb, void *dat) { + unsigned char *tmp, maskAND, maskOR_msb, maskOR_lsb; + int res, err, bsize, maskOR_msb_offset; + + /* sanity check the input */ + if ((size <= 1) || (t <= 0)) { + return MP_VAL; + } + + /* LTM_PRIME_SAFE implies LTM_PRIME_BBS */ + if ((flags & LTM_PRIME_SAFE) != 0) { + flags |= LTM_PRIME_BBS; + } + + /* calc the byte size */ + bsize = (size >> 3) + ((size & 7) ? 1 : 0); + + /* we need a buffer of bsize bytes */ + tmp = OPT_CAST(unsigned char) XMALLOC(bsize); + if (tmp == NULL) { + return MP_MEM; + } + + /* calc the maskAND value for the MSbyte*/ + maskAND = ((size & 7) == 0) ? 0xFF : (0xFF >> (8 - (size & 7))); + + /* calc the maskOR_msb */ + maskOR_msb = 0; + maskOR_msb_offset = ((size & 7) == 1) ? 1 : 0; + if ((flags & LTM_PRIME_2MSB_ON) != 0) { + maskOR_msb |= 0x80 >> ((9 - size) & 7); + } + + /* get the maskOR_lsb */ + maskOR_lsb = 1; + if ((flags & LTM_PRIME_BBS) != 0) { + maskOR_lsb |= 3; + } + + do { + /* read the bytes */ + if (cb(tmp, bsize, dat) != bsize) { + err = MP_VAL; + goto error; + } + + /* work over the MSbyte */ + tmp[0] &= maskAND; + tmp[0] |= 1 << ((size - 1) & 7); + + /* mix in the maskORs */ + tmp[maskOR_msb_offset] |= maskOR_msb; + tmp[bsize - 1] |= maskOR_lsb; + + /* read it in */ + if ((err = mp_read_unsigned_bin(a, tmp, bsize)) != MP_OKAY) { + goto error; + } + + /* is it prime? */ + if ((err = mp_prime_is_prime(a, t, &res)) != MP_OKAY) { + goto error; + } + if (res == MP_NO) { + continue; + } + + if ((flags & LTM_PRIME_SAFE) != 0) { + /* see if (a-1)/2 is prime */ + if ((err = mp_sub_d(a, 1, a)) != MP_OKAY) { + goto error; + } + if ((err = mp_div_2(a, a)) != MP_OKAY) { + goto error; + } + + /* is it prime? */ + if ((err = mp_prime_is_prime(a, t, &res)) != MP_OKAY) { + goto error; + } + } + } while (res == MP_NO); + + if ((flags & LTM_PRIME_SAFE) != 0) { + /* restore a to the original value */ + if ((err = mp_mul_2(a, a)) != MP_OKAY) { + goto error; + } + if ((err = mp_add_d(a, 1, a)) != MP_OKAY) { + goto error; + } + } + + err = MP_OKAY; +error: + XFREE(tmp); + return err; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_RADIX_SIZE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* returns size of ASCII reprensentation */ +int mp_radix_size(mp_int *a, int radix, int *size) { + int res, digs; + mp_int t; + mp_digit d; + + *size = 0; + + /* make sure the radix is in range */ + if ((radix < 2) || (radix > 64)) { + return MP_VAL; + } + + if (mp_iszero(a) == MP_YES) { + *size = 2; + return MP_OKAY; + } + + /* special case for binary */ + if (radix == 2) { + *size = mp_count_bits(a) + ((a->sign == MP_NEG) ? 1 : 0) + 1; + return MP_OKAY; + } + + /* digs is the digit count */ + digs = 0; + + /* if it's negative add one for the sign */ + if (a->sign == MP_NEG) { + ++digs; + } + + /* init a copy of the input */ + if ((res = mp_init_copy(&t, a)) != MP_OKAY) { + return res; + } + + /* force temp to positive */ + t.sign = MP_ZPOS; + + /* fetch out all of the digits */ + while (mp_iszero(&t) == MP_NO) { + if ((res = mp_div_d(&t, (mp_digit)radix, &t, &d)) != MP_OKAY) { + mp_clear(&t); + return res; + } + ++digs; + } + mp_clear(&t); + + /* return digs + 1, the 1 is for the NULL byte that would be required. */ + *size = digs + 1; + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_RADIX_SMAP_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* chars used in radix conversions */ +const char *mp_s_rmap = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz+/"; +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_RAND_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* makes a pseudo-random int of a given size */ +int +mp_rand(mp_int *a, int digits) { + int res; + mp_digit d; + + mp_zero(a); + if (digits <= 0) { + return MP_OKAY; + } + + /* first place a random non-zero digit */ + do { + d = ((mp_digit)abs(MP_GEN_RANDOM())) & MP_MASK; + } while (d == 0); + + if ((res = mp_add_d(a, d, a)) != MP_OKAY) { + return res; + } + + while (--digits > 0) { + if ((res = mp_lshd(a, 1)) != MP_OKAY) { + return res; + } + + if ((res = mp_add_d(a, ((mp_digit)abs(MP_GEN_RANDOM())), a)) != MP_OKAY) { + return res; + } + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_READ_RADIX_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* read a string [ASCII] in a given radix */ +int mp_read_radix(mp_int *a, const char *str, int radix) { + int y, res, neg; + char ch; + + /* zero the digit bignum */ + mp_zero(a); + + /* make sure the radix is ok */ + if ((radix < 2) || (radix > 64)) { + return MP_VAL; + } + + /* if the leading digit is a + * minus set the sign to negative. + */ + if (*str == '-') { + ++str; + neg = MP_NEG; + } else { + neg = MP_ZPOS; + } + + /* set the integer to the default of zero */ + mp_zero(a); + + /* process each digit of the string */ + while (*str != '\0') { + /* if the radix <= 36 the conversion is case insensitive + * this allows numbers like 1AB and 1ab to represent the same value + * [e.g. in hex] + */ + ch = (radix <= 36) ? (char)toupper((int)*str) : *str; + for (y = 0; y < 64; y++) { + if (ch == mp_s_rmap[y]) { + break; + } + } + + /* if the char was found in the map + * and is less than the given radix add it + * to the number, otherwise exit the loop. + */ + if (y < radix) { + if ((res = mp_mul_d(a, (mp_digit)radix, a)) != MP_OKAY) { + return res; + } + if ((res = mp_add_d(a, (mp_digit)y, a)) != MP_OKAY) { + return res; + } + } else { + break; + } + ++str; + } + + /* set the sign only if a != 0 */ + if (mp_iszero(a) != MP_YES) { + a->sign = neg; + } + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_READ_SIGNED_BIN_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* read signed bin, big endian, first byte is 0==positive or 1==negative */ +int mp_read_signed_bin(mp_int *a, const unsigned char *b, int c) { + int res; + + /* read magnitude */ + if ((res = mp_read_unsigned_bin(a, b + 1, c - 1)) != MP_OKAY) { + return res; + } + + /* first byte is 0 for positive, non-zero for negative */ + if (b[0] == 0) { + a->sign = MP_ZPOS; + } else { + a->sign = MP_NEG; + } + + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_READ_UNSIGNED_BIN_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* reads a unsigned char array, assumes the msb is stored first [big endian] */ +int mp_read_unsigned_bin(mp_int *a, const unsigned char *b, int c) { + int res; + + /* make sure there are at least two digits */ + if (a->alloc < 2) { + if ((res = mp_grow(a, 2)) != MP_OKAY) { + return res; + } + } + + /* zero the int */ + mp_zero(a); + + /* read the bytes in */ + while (c-- > 0) { + if ((res = mp_mul_2d(a, 8, a)) != MP_OKAY) { + return res; + } + + #ifndef MP_8BIT + a->dp[0] |= *b++; + a->used += 1; + #else + a->dp[0] = (*b & MP_MASK); + a->dp[1] |= ((*b++ >> 7U) & 1); + a->used += 2; + #endif + } + mp_clamp(a); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_REDUCE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* reduces x mod m, assumes 0 < x < m**2, mu is + * precomputed via mp_reduce_setup. + * From HAC pp.604 Algorithm 14.42 + */ +int mp_reduce(mp_int *x, mp_int *m, mp_int *mu) { + mp_int q; + int res, um = m->used; + + /* q = x */ + if ((res = mp_init_copy(&q, x)) != MP_OKAY) { + return res; + } + + /* q1 = x / b**(k-1) */ + mp_rshd(&q, um - 1); + + /* according to HAC this optimization is ok */ + if (((mp_digit)um) > (((mp_digit)1) << (DIGIT_BIT - 1))) { + if ((res = mp_mul(&q, mu, &q)) != MP_OKAY) { + goto CLEANUP; + } + } else { + #ifdef BN_S_MP_MUL_HIGH_DIGS_C + if ((res = s_mp_mul_high_digs(&q, mu, &q, um)) != MP_OKAY) { + goto CLEANUP; + } + #elif defined(BN_FAST_S_MP_MUL_HIGH_DIGS_C) + if ((res = fast_s_mp_mul_high_digs(&q, mu, &q, um)) != MP_OKAY) { + goto CLEANUP; + } + #else + { + res = MP_VAL; + goto CLEANUP; + } + #endif + } + + /* q3 = q2 / b**(k+1) */ + mp_rshd(&q, um + 1); + + /* x = x mod b**(k+1), quick (no division) */ + if ((res = mp_mod_2d(x, DIGIT_BIT * (um + 1), x)) != MP_OKAY) { + goto CLEANUP; + } + + /* q = q * m mod b**(k+1), quick (no division) */ + if ((res = s_mp_mul_digs(&q, m, &q, um + 1)) != MP_OKAY) { + goto CLEANUP; + } + + /* x = x - q */ + if ((res = mp_sub(x, &q, x)) != MP_OKAY) { + goto CLEANUP; + } + + /* If x < 0, add b**(k+1) to it */ + if (mp_cmp_d(x, 0) == MP_LT) { + mp_set(&q, 1); + if ((res = mp_lshd(&q, um + 1)) != MP_OKAY) + goto CLEANUP; + if ((res = mp_add(x, &q, x)) != MP_OKAY) + goto CLEANUP; + } + + /* Back off if it's too big */ + while (mp_cmp(x, m) != MP_LT) { + if ((res = s_mp_sub(x, m, x)) != MP_OKAY) { + goto CLEANUP; + } + } + +CLEANUP: + mp_clear(&q); + + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_REDUCE_2K_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* reduces a modulo n where n is of the form 2**p - d */ +int mp_reduce_2k(mp_int *a, mp_int *n, mp_digit d) { + mp_int q; + int p, res; + + if ((res = mp_init(&q)) != MP_OKAY) { + return res; + } + + p = mp_count_bits(n); +top: + /* q = a/2**p, a = a mod 2**p */ + if ((res = mp_div_2d(a, p, &q, a)) != MP_OKAY) { + goto ERR; + } + + if (d != 1) { + /* q = q * d */ + if ((res = mp_mul_d(&q, d, &q)) != MP_OKAY) { + goto ERR; + } + } + + /* a = a + q */ + if ((res = s_mp_add(a, &q, a)) != MP_OKAY) { + goto ERR; + } + + if (mp_cmp_mag(a, n) != MP_LT) { + if ((res = s_mp_sub(a, n, a)) != MP_OKAY) { + goto ERR; + } + goto top; + } + +ERR: + mp_clear(&q); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_REDUCE_2K_L_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* reduces a modulo n where n is of the form 2**p - d + This differs from reduce_2k since "d" can be larger + than a single digit. + */ +int mp_reduce_2k_l(mp_int *a, mp_int *n, mp_int *d) { + mp_int q; + int p, res; + + if ((res = mp_init(&q)) != MP_OKAY) { + return res; + } + + p = mp_count_bits(n); +top: + /* q = a/2**p, a = a mod 2**p */ + if ((res = mp_div_2d(a, p, &q, a)) != MP_OKAY) { + goto ERR; + } + + /* q = q * d */ + if ((res = mp_mul(&q, d, &q)) != MP_OKAY) { + goto ERR; + } + + /* a = a + q */ + if ((res = s_mp_add(a, &q, a)) != MP_OKAY) { + goto ERR; + } + + if (mp_cmp_mag(a, n) != MP_LT) { + if ((res = s_mp_sub(a, n, a)) != MP_OKAY) { + goto ERR; + } + goto top; + } + +ERR: + mp_clear(&q); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_REDUCE_2K_SETUP_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* determines the setup value */ +int mp_reduce_2k_setup(mp_int *a, mp_digit *d) { + int res, p; + mp_int tmp; + + if ((res = mp_init(&tmp)) != MP_OKAY) { + return res; + } + + p = mp_count_bits(a); + if ((res = mp_2expt(&tmp, p)) != MP_OKAY) { + mp_clear(&tmp); + return res; + } + + if ((res = s_mp_sub(&tmp, a, &tmp)) != MP_OKAY) { + mp_clear(&tmp); + return res; + } + + *d = tmp.dp[0]; + mp_clear(&tmp); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_REDUCE_2K_SETUP_L_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* determines the setup value */ +int mp_reduce_2k_setup_l(mp_int *a, mp_int *d) { + int res; + mp_int tmp; + + if ((res = mp_init(&tmp)) != MP_OKAY) { + return res; + } + + if ((res = mp_2expt(&tmp, mp_count_bits(a))) != MP_OKAY) { + goto ERR; + } + + if ((res = s_mp_sub(&tmp, a, d)) != MP_OKAY) { + goto ERR; + } + +ERR: + mp_clear(&tmp); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_REDUCE_IS_2K_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* determines if mp_reduce_2k can be used */ +int mp_reduce_is_2k(mp_int *a) { + int ix, iy, iw; + mp_digit iz; + + if (a->used == 0) { + return MP_NO; + } else if (a->used == 1) { + return MP_YES; + } else if (a->used > 1) { + iy = mp_count_bits(a); + iz = 1; + iw = 1; + + /* Test every bit from the second digit up, must be 1 */ + for (ix = DIGIT_BIT; ix < iy; ix++) { + if ((a->dp[iw] & iz) == 0) { + return MP_NO; + } + iz <<= 1; + if (iz > (mp_digit)MP_MASK) { + ++iw; + iz = 1; + } + } + } + return MP_YES; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_REDUCE_IS_2K_L_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* determines if reduce_2k_l can be used */ +int mp_reduce_is_2k_l(mp_int *a) { + int ix, iy; + + if (a->used == 0) { + return MP_NO; + } else if (a->used == 1) { + return MP_YES; + } else if (a->used > 1) { + /* if more than half of the digits are -1 we're sold */ + for (iy = ix = 0; ix < a->used; ix++) { + if (a->dp[ix] == MP_MASK) { + ++iy; + } + } + return (iy >= (a->used / 2)) ? MP_YES : MP_NO; + } + return MP_NO; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_REDUCE_SETUP_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* pre-calculate the value required for Barrett reduction + * For a given modulus "b" it calulates the value required in "a" + */ +int mp_reduce_setup(mp_int *a, mp_int *b) { + int res; + + if ((res = mp_2expt(a, b->used * 2 * DIGIT_BIT)) != MP_OKAY) { + return res; + } + return mp_div(a, b, a, NULL); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_RSHD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* shift right a certain amount of digits */ +void mp_rshd(mp_int *a, int b) { + int x; + + /* if b <= 0 then ignore it */ + if (b <= 0) { + return; + } + + /* if b > used then simply zero it and return */ + if (a->used <= b) { + mp_zero(a); + return; + } + + { + mp_digit *bottom, *top; + + /* shift the digits down */ + + /* bottom */ + bottom = a->dp; + + /* top [offset into digits] */ + top = a->dp + b; + + /* this is implemented as a sliding window where + * the window is b-digits long and digits from + * the top of the window are copied to the bottom + * + * e.g. + + b-2 | b-1 | b0 | b1 | b2 | ... | bb | ----> + /\ | ----> + **\-------------------/ ----> + */ + for (x = 0; x < (a->used - b); x++) { + *bottom++ = *top++; + } + + /* zero the top digits */ + for ( ; x < a->used; x++) { + *bottom++ = 0; + } + } + + /* remove excess digits */ + a->used -= b; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SET_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* set to a digit */ +void mp_set(mp_int *a, mp_digit b) { + mp_zero(a); + a->dp[0] = b & MP_MASK; + a->used = (a->dp[0] != 0) ? 1 : 0; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SET_INT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* set a 32-bit const */ +int mp_set_int(mp_int *a, unsigned long b) { + int x, res; + + mp_zero(a); + + /* set four bits at a time */ + for (x = 0; x < 8; x++) { + /* shift the number up four bits */ + if ((res = mp_mul_2d(a, 4, a)) != MP_OKAY) { + return res; + } + + /* OR in the top four bits of the source */ + a->dp[0] |= (b >> 28) & 15; + + /* shift the source up to the next four bits */ + b <<= 4; + + /* ensure that digits are not clamped off */ + a->used += 1; + } + mp_clamp(a); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SET_LONG_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* set a platform dependent unsigned long int */ +MP_SET_XLONG(mp_set_long, unsigned long) +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SET_LONG_LONG_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* set a platform dependent unsigned long long int */ +MP_SET_XLONG(mp_set_long_long, unsigned long long) +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SHRINK_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* shrink a bignum */ +int mp_shrink(mp_int *a) { + mp_digit *tmp; + int used = 1; + + if (a->used > 0) { + used = a->used; + } + + if (a->alloc != used) { + if ((tmp = OPT_CAST(mp_digit) XREALLOC(a->dp, sizeof(mp_digit) * used)) == NULL) { + return MP_MEM; + } + a->dp = tmp; + a->alloc = used; + } + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SIGNED_BIN_SIZE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* get the size for an signed equivalent */ +int mp_signed_bin_size(mp_int *a) { + return 1 + mp_unsigned_bin_size(a); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SQR_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* computes b = a*a */ +int +mp_sqr(mp_int *a, mp_int *b) { + int res; + + #ifdef BN_MP_TOOM_SQR_C + /* use Toom-Cook? */ + if (a->used >= TOOM_SQR_CUTOFF) { + res = mp_toom_sqr(a, b); + /* Karatsuba? */ + } else + #endif + #ifdef BN_MP_KARATSUBA_SQR_C + if (a->used >= KARATSUBA_SQR_CUTOFF) { + res = mp_karatsuba_sqr(a, b); + } else + #endif + { + #ifdef BN_FAST_S_MP_SQR_C + /* can we use the fast comba multiplier? */ + if ((((a->used * 2) + 1) < MP_WARRAY) && + (a->used < + (1 << (((sizeof(mp_word) * CHAR_BIT) - (2 * DIGIT_BIT)) - 1)))) { + res = fast_s_mp_sqr(a, b); + } else + #endif + { + #ifdef BN_S_MP_SQR_C + res = s_mp_sqr(a, b); + #else + res = MP_VAL; + #endif + } + } + b->sign = MP_ZPOS; + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SQRMOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* c = a * a (mod b) */ +int +mp_sqrmod(mp_int *a, mp_int *b, mp_int *c) { + int res; + mp_int t; + + if ((res = mp_init(&t)) != MP_OKAY) { + return res; + } + + if ((res = mp_sqr(a, &t)) != MP_OKAY) { + mp_clear(&t); + return res; + } + res = mp_mod(&t, b, c); + mp_clear(&t); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SQRT_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* this function is less generic than mp_n_root, simpler and faster */ +int mp_sqrt(mp_int *arg, mp_int *ret) { + int res; + mp_int t1, t2; + + /* must be positive */ + if (arg->sign == MP_NEG) { + return MP_VAL; + } + + /* easy out */ + if (mp_iszero(arg) == MP_YES) { + mp_zero(ret); + return MP_OKAY; + } + + if ((res = mp_init_copy(&t1, arg)) != MP_OKAY) { + return res; + } + + if ((res = mp_init(&t2)) != MP_OKAY) { + goto E2; + } + + /* First approx. (not very bad for large arg) */ + mp_rshd(&t1, t1.used / 2); + + /* t1 > 0 */ + if ((res = mp_div(arg, &t1, &t2, NULL)) != MP_OKAY) { + goto E1; + } + if ((res = mp_add(&t1, &t2, &t1)) != MP_OKAY) { + goto E1; + } + if ((res = mp_div_2(&t1, &t1)) != MP_OKAY) { + goto E1; + } + /* And now t1 > sqrt(arg) */ + do { + if ((res = mp_div(arg, &t1, &t2, NULL)) != MP_OKAY) { + goto E1; + } + if ((res = mp_add(&t1, &t2, &t1)) != MP_OKAY) { + goto E1; + } + if ((res = mp_div_2(&t1, &t1)) != MP_OKAY) { + goto E1; + } + /* t1 >= sqrt(arg) >= t2 at this point */ + } while (mp_cmp_mag(&t1, &t2) == MP_GT); + + mp_exch(&t1, ret); + +E1: mp_clear(&t2); +E2: mp_clear(&t1); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SQRTMOD_PRIME_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library is free for all purposes without any express + * guarantee it works. + */ + +/* Tonelli-Shanks algorithm + * https://en.wikipedia.org/wiki/Tonelli%E2%80%93Shanks_algorithm + * https://gmplib.org/list-archives/gmp-discuss/2013-April/005300.html + * + */ + +int mp_sqrtmod_prime(mp_int *n, mp_int *prime, mp_int *ret) { + int res, legendre; + mp_int t1, C, Q, S, Z, M, T, R, two; + mp_digit i; + + /* first handle the simple cases */ + if (mp_cmp_d(n, 0) == MP_EQ) { + mp_zero(ret); + return MP_OKAY; + } + if (mp_cmp_d(prime, 2) == MP_EQ) return MP_VAL; /* prime must be odd */ + if ((res = mp_jacobi(n, prime, &legendre)) != MP_OKAY) return res; + if (legendre == -1) return MP_VAL; /* quadratic non-residue mod prime */ + + if ((res = mp_init_multi(&t1, &C, &Q, &S, &Z, &M, &T, &R, &two, NULL)) != MP_OKAY) { + return res; + } + + /* SPECIAL CASE: if prime mod 4 == 3 + * compute directly: res = n^(prime+1)/4 mod prime + * Handbook of Applied Cryptography algorithm 3.36 + */ + if ((res = mp_mod_d(prime, 4, &i)) != MP_OKAY) goto cleanup; + if (i == 3) { + if ((res = mp_add_d(prime, 1, &t1)) != MP_OKAY) goto cleanup; + if ((res = mp_div_2(&t1, &t1)) != MP_OKAY) goto cleanup; + if ((res = mp_div_2(&t1, &t1)) != MP_OKAY) goto cleanup; + if ((res = mp_exptmod(n, &t1, prime, ret)) != MP_OKAY) goto cleanup; + res = MP_OKAY; + goto cleanup; + } + + /* NOW: Tonelli-Shanks algorithm */ + + /* factor out powers of 2 from prime-1, defining Q and S as: prime-1 = Q*2^S */ + if ((res = mp_copy(prime, &Q)) != MP_OKAY) goto cleanup; + if ((res = mp_sub_d(&Q, 1, &Q)) != MP_OKAY) goto cleanup; + /* Q = prime - 1 */ + mp_zero(&S); + /* S = 0 */ + while (mp_iseven(&Q) != MP_NO) { + if ((res = mp_div_2(&Q, &Q)) != MP_OKAY) goto cleanup; + /* Q = Q / 2 */ + if ((res = mp_add_d(&S, 1, &S)) != MP_OKAY) goto cleanup; + /* S = S + 1 */ + } + + /* find a Z such that the Legendre symbol (Z|prime) == -1 */ + if ((res = mp_set_int(&Z, 2)) != MP_OKAY) goto cleanup; + /* Z = 2 */ + while (1) { + if ((res = mp_jacobi(&Z, prime, &legendre)) != MP_OKAY) goto cleanup; + if (legendre == -1) break; + if ((res = mp_add_d(&Z, 1, &Z)) != MP_OKAY) goto cleanup; + /* Z = Z + 1 */ + } + + if ((res = mp_exptmod(&Z, &Q, prime, &C)) != MP_OKAY) goto cleanup; + /* C = Z ^ Q mod prime */ + if ((res = mp_add_d(&Q, 1, &t1)) != MP_OKAY) goto cleanup; + if ((res = mp_div_2(&t1, &t1)) != MP_OKAY) goto cleanup; + /* t1 = (Q + 1) / 2 */ + if ((res = mp_exptmod(n, &t1, prime, &R)) != MP_OKAY) goto cleanup; + /* R = n ^ ((Q + 1) / 2) mod prime */ + if ((res = mp_exptmod(n, &Q, prime, &T)) != MP_OKAY) goto cleanup; + /* T = n ^ Q mod prime */ + if ((res = mp_copy(&S, &M)) != MP_OKAY) goto cleanup; + /* M = S */ + if ((res = mp_set_int(&two, 2)) != MP_OKAY) goto cleanup; + + res = MP_VAL; + while (1) { + if ((res = mp_copy(&T, &t1)) != MP_OKAY) goto cleanup; + i = 0; + while (1) { + if (mp_cmp_d(&t1, 1) == MP_EQ) break; + if ((res = mp_exptmod(&t1, &two, prime, &t1)) != MP_OKAY) goto cleanup; + i++; + } + if (i == 0) { + if ((res = mp_copy(&R, ret)) != MP_OKAY) goto cleanup; + res = MP_OKAY; + goto cleanup; + } + if ((res = mp_sub_d(&M, i, &t1)) != MP_OKAY) goto cleanup; + if ((res = mp_sub_d(&t1, 1, &t1)) != MP_OKAY) goto cleanup; + if ((res = mp_exptmod(&two, &t1, prime, &t1)) != MP_OKAY) goto cleanup; + /* t1 = 2 ^ (M - i - 1) */ + if ((res = mp_exptmod(&C, &t1, prime, &t1)) != MP_OKAY) goto cleanup; + /* t1 = C ^ (2 ^ (M - i - 1)) mod prime */ + if ((res = mp_sqrmod(&t1, prime, &C)) != MP_OKAY) goto cleanup; + /* C = (t1 * t1) mod prime */ + if ((res = mp_mulmod(&R, &t1, prime, &R)) != MP_OKAY) goto cleanup; + /* R = (R * t1) mod prime */ + if ((res = mp_mulmod(&T, &C, prime, &T)) != MP_OKAY) goto cleanup; + /* T = (T * C) mod prime */ + mp_set(&M, i); + /* M = i */ + } + +cleanup: + mp_clear_multi(&t1, &C, &Q, &S, &Z, &M, &T, &R, &two, NULL); + return res; +} +#endif + + + +#ifdef BN_MP_SUB_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* high level subtraction (handles signs) */ +int +mp_sub(mp_int *a, mp_int *b, mp_int *c) { + int sa, sb, res; + + sa = a->sign; + sb = b->sign; + + if (sa != sb) { + /* subtract a negative from a positive, OR */ + /* subtract a positive from a negative. */ + /* In either case, ADD their magnitudes, */ + /* and use the sign of the first number. */ + c->sign = sa; + res = s_mp_add(a, b, c); + } else { + /* subtract a positive from a positive, OR */ + /* subtract a negative from a negative. */ + /* First, take the difference between their */ + /* magnitudes, then... */ + if (mp_cmp_mag(a, b) != MP_LT) { + /* Copy the sign from the first */ + c->sign = sa; + /* The first has a larger or equal magnitude */ + res = s_mp_sub(a, b, c); + } else { + /* The result has the *opposite* sign from */ + /* the first number. */ + c->sign = (sa == MP_ZPOS) ? MP_NEG : MP_ZPOS; + /* The second has a larger magnitude */ + res = s_mp_sub(b, a, c); + } + } + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SUB_D_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* single digit subtraction */ +int +mp_sub_d(mp_int *a, mp_digit b, mp_int *c) { + mp_digit *tmpa, *tmpc, mu; + int res, ix, oldused; + + /* grow c as required */ + if (c->alloc < (a->used + 1)) { + if ((res = mp_grow(c, a->used + 1)) != MP_OKAY) { + return res; + } + } + + /* if a is negative just do an unsigned + * addition [with fudged signs] + */ + if (a->sign == MP_NEG) { + a->sign = MP_ZPOS; + res = mp_add_d(a, b, c); + a->sign = c->sign = MP_NEG; + + /* clamp */ + mp_clamp(c); + + return res; + } + + /* setup regs */ + oldused = c->used; + tmpa = a->dp; + tmpc = c->dp; + + /* if a <= b simply fix the single digit */ + if (((a->used == 1) && (a->dp[0] <= b)) || (a->used == 0)) { + if (a->used == 1) { + *tmpc++ = b - *tmpa; + } else { + *tmpc++ = b; + } + ix = 1; + + /* negative/1digit */ + c->sign = MP_NEG; + c->used = 1; + } else { + /* positive/size */ + c->sign = MP_ZPOS; + c->used = a->used; + + /* subtract first digit */ + *tmpc = *tmpa++ - b; + mu = *tmpc >> ((sizeof(mp_digit) * CHAR_BIT) - 1); + *tmpc++ &= MP_MASK; + + /* handle rest of the digits */ + for (ix = 1; ix < a->used; ix++) { + *tmpc = *tmpa++ - mu; + mu = *tmpc >> ((sizeof(mp_digit) * CHAR_BIT) - 1); + *tmpc++ &= MP_MASK; + } + } + + /* zero excess digits */ + while (ix++ < oldused) { + *tmpc++ = 0; + } + mp_clamp(c); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_SUBMOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* d = a - b (mod c) */ +int +mp_submod(mp_int *a, mp_int *b, mp_int *c, mp_int *d) { + int res; + mp_int t; + + + if ((res = mp_init(&t)) != MP_OKAY) { + return res; + } + + if ((res = mp_sub(a, b, &t)) != MP_OKAY) { + mp_clear(&t); + return res; + } + res = mp_mod(&t, c, d); + mp_clear(&t); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_TO_SIGNED_BIN_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* store in signed [big endian] format */ +int mp_to_signed_bin(mp_int *a, unsigned char *b) { + int res; + + if ((res = mp_to_unsigned_bin(a, b + 1)) != MP_OKAY) { + return res; + } + b[0] = (a->sign == MP_ZPOS) ? (unsigned char)0 : (unsigned char)1; + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_TO_SIGNED_BIN_N_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* store in signed [big endian] format */ +int mp_to_signed_bin_n(mp_int *a, unsigned char *b, unsigned long *outlen) { + if (*outlen < (unsigned long)mp_signed_bin_size(a)) { + return MP_VAL; + } + *outlen = mp_signed_bin_size(a); + return mp_to_signed_bin(a, b); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_TO_UNSIGNED_BIN_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* store in unsigned [big endian] format */ +int mp_to_unsigned_bin(mp_int *a, unsigned char *b) { + int x, res; + mp_int t; + + if ((res = mp_init_copy(&t, a)) != MP_OKAY) { + return res; + } + + x = 0; + while (mp_iszero(&t) == MP_NO) { + #ifndef MP_8BIT + b[x++] = (unsigned char)(t.dp[0] & 255); + #else + b[x++] = (unsigned char)(t.dp[0] | ((t.dp[1] & 0x01) << 7)); + #endif + if ((res = mp_div_2d(&t, 8, &t, NULL)) != MP_OKAY) { + mp_clear(&t); + return res; + } + } + bn_reverse(b, x); + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_TOOM_MUL_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* multiplication using the Toom-Cook 3-way algorithm + * + * Much more complicated than Karatsuba but has a lower + * asymptotic running time of O(N**1.464). This algorithm is + * only particularly useful on VERY large inputs + * (we're talking 1000s of digits here...). + */ +int mp_toom_mul(mp_int *a, mp_int *b, mp_int *c) { + mp_int w0, w1, w2, w3, w4, tmp1, tmp2, a0, a1, a2, b0, b1, b2; + int res, B; + + /* init temps */ + if ((res = mp_init_multi(&w0, &w1, &w2, &w3, &w4, + &a0, &a1, &a2, &b0, &b1, + &b2, &tmp1, &tmp2, NULL)) != MP_OKAY) { + return res; + } + + /* B */ + B = MIN(a->used, b->used) / 3; + + /* a = a2 * B**2 + a1 * B + a0 */ + if ((res = mp_mod_2d(a, DIGIT_BIT * B, &a0)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_copy(a, &a1)) != MP_OKAY) { + goto ERR; + } + mp_rshd(&a1, B); + if ((res = mp_mod_2d(&a1, DIGIT_BIT * B, &a1)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_copy(a, &a2)) != MP_OKAY) { + goto ERR; + } + mp_rshd(&a2, B * 2); + + /* b = b2 * B**2 + b1 * B + b0 */ + if ((res = mp_mod_2d(b, DIGIT_BIT * B, &b0)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_copy(b, &b1)) != MP_OKAY) { + goto ERR; + } + mp_rshd(&b1, B); + (void)mp_mod_2d(&b1, DIGIT_BIT * B, &b1); + + if ((res = mp_copy(b, &b2)) != MP_OKAY) { + goto ERR; + } + mp_rshd(&b2, B * 2); + + /* w0 = a0*b0 */ + if ((res = mp_mul(&a0, &b0, &w0)) != MP_OKAY) { + goto ERR; + } + + /* w4 = a2 * b2 */ + if ((res = mp_mul(&a2, &b2, &w4)) != MP_OKAY) { + goto ERR; + } + + /* w1 = (a2 + 2(a1 + 2a0))(b2 + 2(b1 + 2b0)) */ + if ((res = mp_mul_2(&a0, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_mul_2(&tmp1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a2, &tmp1)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_mul_2(&b0, &tmp2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp2, &b1, &tmp2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_mul_2(&tmp2, &tmp2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp2, &b2, &tmp2)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_mul(&tmp1, &tmp2, &w1)) != MP_OKAY) { + goto ERR; + } + + /* w3 = (a0 + 2(a1 + 2a2))(b0 + 2(b1 + 2b2)) */ + if ((res = mp_mul_2(&a2, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_mul_2(&tmp1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a0, &tmp1)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_mul_2(&b2, &tmp2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp2, &b1, &tmp2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_mul_2(&tmp2, &tmp2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp2, &b0, &tmp2)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_mul(&tmp1, &tmp2, &w3)) != MP_OKAY) { + goto ERR; + } + + + /* w2 = (a2 + a1 + a0)(b2 + b1 + b0) */ + if ((res = mp_add(&a2, &a1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a0, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&b2, &b1, &tmp2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp2, &b0, &tmp2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_mul(&tmp1, &tmp2, &w2)) != MP_OKAY) { + goto ERR; + } + + /* now solve the matrix + + 0 0 0 0 1 + 1 2 4 8 16 + 1 1 1 1 1 + 16 8 4 2 1 + 1 0 0 0 0 + + using 12 subtractions, 4 shifts, + 2 small divisions and 1 small multiplication + */ + + /* r1 - r4 */ + if ((res = mp_sub(&w1, &w4, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3 - r0 */ + if ((res = mp_sub(&w3, &w0, &w3)) != MP_OKAY) { + goto ERR; + } + /* r1/2 */ + if ((res = mp_div_2(&w1, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3/2 */ + if ((res = mp_div_2(&w3, &w3)) != MP_OKAY) { + goto ERR; + } + /* r2 - r0 - r4 */ + if ((res = mp_sub(&w2, &w0, &w2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w2, &w4, &w2)) != MP_OKAY) { + goto ERR; + } + /* r1 - r2 */ + if ((res = mp_sub(&w1, &w2, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3 - r2 */ + if ((res = mp_sub(&w3, &w2, &w3)) != MP_OKAY) { + goto ERR; + } + /* r1 - 8r0 */ + if ((res = mp_mul_2d(&w0, 3, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w1, &tmp1, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3 - 8r4 */ + if ((res = mp_mul_2d(&w4, 3, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w3, &tmp1, &w3)) != MP_OKAY) { + goto ERR; + } + /* 3r2 - r1 - r3 */ + if ((res = mp_mul_d(&w2, 3, &w2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w2, &w1, &w2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w2, &w3, &w2)) != MP_OKAY) { + goto ERR; + } + /* r1 - r2 */ + if ((res = mp_sub(&w1, &w2, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3 - r2 */ + if ((res = mp_sub(&w3, &w2, &w3)) != MP_OKAY) { + goto ERR; + } + /* r1/3 */ + if ((res = mp_div_3(&w1, &w1, NULL)) != MP_OKAY) { + goto ERR; + } + /* r3/3 */ + if ((res = mp_div_3(&w3, &w3, NULL)) != MP_OKAY) { + goto ERR; + } + + /* at this point shift W[n] by B*n */ + if ((res = mp_lshd(&w1, 1 * B)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_lshd(&w2, 2 * B)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_lshd(&w3, 3 * B)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_lshd(&w4, 4 * B)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_add(&w0, &w1, c)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&w2, &w3, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&w4, &tmp1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, c, c)) != MP_OKAY) { + goto ERR; + } + +ERR: + mp_clear_multi(&w0, &w1, &w2, &w3, &w4, + &a0, &a1, &a2, &b0, &b1, + &b2, &tmp1, &tmp2, NULL); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_TOOM_SQR_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* squaring using Toom-Cook 3-way algorithm */ +int +mp_toom_sqr(mp_int *a, mp_int *b) { + mp_int w0, w1, w2, w3, w4, tmp1, a0, a1, a2; + int res, B; + + /* init temps */ + if ((res = mp_init_multi(&w0, &w1, &w2, &w3, &w4, &a0, &a1, &a2, &tmp1, NULL)) != MP_OKAY) { + return res; + } + + /* B */ + B = a->used / 3; + + /* a = a2 * B**2 + a1 * B + a0 */ + if ((res = mp_mod_2d(a, DIGIT_BIT * B, &a0)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_copy(a, &a1)) != MP_OKAY) { + goto ERR; + } + mp_rshd(&a1, B); + if ((res = mp_mod_2d(&a1, DIGIT_BIT * B, &a1)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_copy(a, &a2)) != MP_OKAY) { + goto ERR; + } + mp_rshd(&a2, B * 2); + + /* w0 = a0*a0 */ + if ((res = mp_sqr(&a0, &w0)) != MP_OKAY) { + goto ERR; + } + + /* w4 = a2 * a2 */ + if ((res = mp_sqr(&a2, &w4)) != MP_OKAY) { + goto ERR; + } + + /* w1 = (a2 + 2(a1 + 2a0))**2 */ + if ((res = mp_mul_2(&a0, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_mul_2(&tmp1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a2, &tmp1)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_sqr(&tmp1, &w1)) != MP_OKAY) { + goto ERR; + } + + /* w3 = (a0 + 2(a1 + 2a2))**2 */ + if ((res = mp_mul_2(&a2, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_mul_2(&tmp1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a0, &tmp1)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_sqr(&tmp1, &w3)) != MP_OKAY) { + goto ERR; + } + + + /* w2 = (a2 + a1 + a0)**2 */ + if ((res = mp_add(&a2, &a1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, &a0, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sqr(&tmp1, &w2)) != MP_OKAY) { + goto ERR; + } + + /* now solve the matrix + + 0 0 0 0 1 + 1 2 4 8 16 + 1 1 1 1 1 + 16 8 4 2 1 + 1 0 0 0 0 + + using 12 subtractions, 4 shifts, 2 small divisions and 1 small multiplication. + */ + + /* r1 - r4 */ + if ((res = mp_sub(&w1, &w4, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3 - r0 */ + if ((res = mp_sub(&w3, &w0, &w3)) != MP_OKAY) { + goto ERR; + } + /* r1/2 */ + if ((res = mp_div_2(&w1, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3/2 */ + if ((res = mp_div_2(&w3, &w3)) != MP_OKAY) { + goto ERR; + } + /* r2 - r0 - r4 */ + if ((res = mp_sub(&w2, &w0, &w2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w2, &w4, &w2)) != MP_OKAY) { + goto ERR; + } + /* r1 - r2 */ + if ((res = mp_sub(&w1, &w2, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3 - r2 */ + if ((res = mp_sub(&w3, &w2, &w3)) != MP_OKAY) { + goto ERR; + } + /* r1 - 8r0 */ + if ((res = mp_mul_2d(&w0, 3, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w1, &tmp1, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3 - 8r4 */ + if ((res = mp_mul_2d(&w4, 3, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w3, &tmp1, &w3)) != MP_OKAY) { + goto ERR; + } + /* 3r2 - r1 - r3 */ + if ((res = mp_mul_d(&w2, 3, &w2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w2, &w1, &w2)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_sub(&w2, &w3, &w2)) != MP_OKAY) { + goto ERR; + } + /* r1 - r2 */ + if ((res = mp_sub(&w1, &w2, &w1)) != MP_OKAY) { + goto ERR; + } + /* r3 - r2 */ + if ((res = mp_sub(&w3, &w2, &w3)) != MP_OKAY) { + goto ERR; + } + /* r1/3 */ + if ((res = mp_div_3(&w1, &w1, NULL)) != MP_OKAY) { + goto ERR; + } + /* r3/3 */ + if ((res = mp_div_3(&w3, &w3, NULL)) != MP_OKAY) { + goto ERR; + } + + /* at this point shift W[n] by B*n */ + if ((res = mp_lshd(&w1, 1 * B)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_lshd(&w2, 2 * B)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_lshd(&w3, 3 * B)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_lshd(&w4, 4 * B)) != MP_OKAY) { + goto ERR; + } + + if ((res = mp_add(&w0, &w1, b)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&w2, &w3, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&w4, &tmp1, &tmp1)) != MP_OKAY) { + goto ERR; + } + if ((res = mp_add(&tmp1, b, b)) != MP_OKAY) { + goto ERR; + } + +ERR: + mp_clear_multi(&w0, &w1, &w2, &w3, &w4, &a0, &a1, &a2, &tmp1, NULL); + return res; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_TORADIX_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* stores a bignum as a ASCII string in a given radix (2..64) */ +int mp_toradix(mp_int *a, char *str, int radix) { + int res, digs; + mp_int t; + mp_digit d; + char *_s = str; + + /* check range of the radix */ + if ((radix < 2) || (radix > 64)) { + return MP_VAL; + } + + /* quick out if its zero */ + if (mp_iszero(a) == MP_YES) { + *str++ = '0'; + *str = '\0'; + return MP_OKAY; + } + + if ((res = mp_init_copy(&t, a)) != MP_OKAY) { + return res; + } + + /* if it is negative output a - */ + if (t.sign == MP_NEG) { + ++_s; + *str++ = '-'; + t.sign = MP_ZPOS; + } + + digs = 0; + while (mp_iszero(&t) == MP_NO) { + if ((res = mp_div_d(&t, (mp_digit)radix, &t, &d)) != MP_OKAY) { + mp_clear(&t); + return res; + } + *str++ = mp_s_rmap[d]; + ++digs; + } + + /* reverse the digits of the string. In this case _s points + * to the first digit [exluding the sign] of the number] + */ + bn_reverse((unsigned char *)_s, digs); + + /* append a NULL so the string is properly terminated */ + *str = '\0'; + + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_UNSIGNED_BIN_SIZE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* get the size for an unsigned equivalent */ +int mp_unsigned_bin_size(mp_int *a) { + int size = mp_count_bits(a); + + return (size / 8) + (((size & 7) != 0) ? 1 : 0); +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_XOR_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* XOR two ints together */ +int +mp_xor(mp_int *a, mp_int *b, mp_int *c) { + int res, ix, px; + mp_int t, *x; + + if (a->used > b->used) { + if ((res = mp_init_copy(&t, a)) != MP_OKAY) { + return res; + } + px = b->used; + x = b; + } else { + if ((res = mp_init_copy(&t, b)) != MP_OKAY) { + return res; + } + px = a->used; + x = a; + } + + for (ix = 0; ix < px; ix++) { + t.dp[ix] ^= x->dp[ix]; + } + mp_clamp(&t); + mp_exch(c, &t); + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_MP_ZERO_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* set to zero */ +void mp_zero(mp_int *a) { + int n; + mp_digit *tmp; + + a->sign = MP_ZPOS; + a->used = 0; + + tmp = a->dp; + for (n = 0; n < a->alloc; n++) { + *tmp++ = 0; + } +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_PRIME_TAB_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ +const mp_digit ltm_prime_tab[] = { + 0x0002, 0x0003, 0x0005, 0x0007, 0x000B, 0x000D, 0x0011, 0x0013, + 0x0017, 0x001D, 0x001F, 0x0025, 0x0029, 0x002B, 0x002F, 0x0035, + 0x003B, 0x003D, 0x0043, 0x0047, 0x0049, 0x004F, 0x0053, 0x0059, + 0x0061, 0x0065, 0x0067, 0x006B, 0x006D, 0x0071, 0x007F, + #ifndef MP_8BIT + 0x0083, + 0x0089, 0x008B, 0x0095, 0x0097, 0x009D, 0x00A3, 0x00A7, 0x00AD, + 0x00B3, 0x00B5, 0x00BF, 0x00C1, 0x00C5, 0x00C7, 0x00D3, 0x00DF, + 0x00E3, 0x00E5, 0x00E9, 0x00EF, 0x00F1, 0x00FB, 0x0101, 0x0107, + 0x010D, 0x010F, 0x0115, 0x0119, 0x011B, 0x0125, 0x0133, 0x0137, + + 0x0139, 0x013D, 0x014B, 0x0151, 0x015B, 0x015D, 0x0161, 0x0167, + 0x016F, 0x0175, 0x017B, 0x017F, 0x0185, 0x018D, 0x0191, 0x0199, + 0x01A3, 0x01A5, 0x01AF, 0x01B1, 0x01B7, 0x01BB, 0x01C1, 0x01C9, + 0x01CD, 0x01CF, 0x01D3, 0x01DF, 0x01E7, 0x01EB, 0x01F3, 0x01F7, + 0x01FD, 0x0209, 0x020B, 0x021D, 0x0223, 0x022D, 0x0233, 0x0239, + 0x023B, 0x0241, 0x024B, 0x0251, 0x0257, 0x0259, 0x025F, 0x0265, + 0x0269, 0x026B, 0x0277, 0x0281, 0x0283, 0x0287, 0x028D, 0x0293, + 0x0295, 0x02A1, 0x02A5, 0x02AB, 0x02B3, 0x02BD, 0x02C5, 0x02CF, + + 0x02D7, 0x02DD, 0x02E3, 0x02E7, 0x02EF, 0x02F5, 0x02F9, 0x0301, + 0x0305, 0x0313, 0x031D, 0x0329, 0x032B, 0x0335, 0x0337, 0x033B, + 0x033D, 0x0347, 0x0355, 0x0359, 0x035B, 0x035F, 0x036D, 0x0371, + 0x0373, 0x0377, 0x038B, 0x038F, 0x0397, 0x03A1, 0x03A9, 0x03AD, + 0x03B3, 0x03B9, 0x03C7, 0x03CB, 0x03D1, 0x03D7, 0x03DF, 0x03E5, + 0x03F1, 0x03F5, 0x03FB, 0x03FD, 0x0407, 0x0409, 0x040F, 0x0419, + 0x041B, 0x0425, 0x0427, 0x042D, 0x043F, 0x0443, 0x0445, 0x0449, + 0x044F, 0x0455, 0x045D, 0x0463, 0x0469, 0x047F, 0x0481, 0x048B, + + 0x0493, 0x049D, 0x04A3, 0x04A9, 0x04B1, 0x04BD, 0x04C1, 0x04C7, + 0x04CD, 0x04CF, 0x04D5, 0x04E1, 0x04EB, 0x04FD, 0x04FF, 0x0503, + 0x0509, 0x050B, 0x0511, 0x0515, 0x0517, 0x051B, 0x0527, 0x0529, + 0x052F, 0x0551, 0x0557, 0x055D, 0x0565, 0x0577, 0x0581, 0x058F, + 0x0593, 0x0595, 0x0599, 0x059F, 0x05A7, 0x05AB, 0x05AD, 0x05B3, + 0x05BF, 0x05C9, 0x05CB, 0x05CF, 0x05D1, 0x05D5, 0x05DB, 0x05E7, + 0x05F3, 0x05FB, 0x0607, 0x060D, 0x0611, 0x0617, 0x061F, 0x0623, + 0x062B, 0x062F, 0x063D, 0x0641, 0x0647, 0x0649, 0x064D, 0x0653 + #endif +}; +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_REVERSE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* reverse an array, used for radix code */ +void +bn_reverse(unsigned char *s, int len) { + int ix, iy; + unsigned char t; + + ix = 0; + iy = len - 1; + while (ix < iy) { + t = s[ix]; + s[ix] = s[iy]; + s[iy] = t; + ++ix; + --iy; + } +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_S_MP_ADD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* low level addition, based on HAC pp.594, Algorithm 14.7 */ +int +s_mp_add(mp_int *a, mp_int *b, mp_int *c) { + mp_int *x; + int olduse, res, min, max; + + /* find sizes, we let |a| <= |b| which means we have to sort + * them. "x" will point to the input with the most digits + */ + if (a->used > b->used) { + min = b->used; + max = a->used; + x = a; + } else { + min = a->used; + max = b->used; + x = b; + } + + /* init result */ + if (c->alloc < (max + 1)) { + if ((res = mp_grow(c, max + 1)) != MP_OKAY) { + return res; + } + } + + /* get old used digit count and set new one */ + olduse = c->used; + c->used = max + 1; + + { + mp_digit u, *tmpa, *tmpb, *tmpc; + int i; + + /* alias for digit pointers */ + + /* first input */ + tmpa = a->dp; + + /* second input */ + tmpb = b->dp; + + /* destination */ + tmpc = c->dp; + + /* zero the carry */ + u = 0; + for (i = 0; i < min; i++) { + /* Compute the sum at one digit, T[i] = A[i] + B[i] + U */ + *tmpc = *tmpa++ + *tmpb++ + u; + + /* U = carry bit of T[i] */ + u = *tmpc >> ((mp_digit)DIGIT_BIT); + + /* take away carry bit from T[i] */ + *tmpc++ &= MP_MASK; + } + + /* now copy higher words if any, that is in A+B + * if A or B has more digits add those in + */ + if (min != max) { + for ( ; i < max; i++) { + /* T[i] = X[i] + U */ + *tmpc = x->dp[i] + u; + + /* U = carry bit of T[i] */ + u = *tmpc >> ((mp_digit)DIGIT_BIT); + + /* take away carry bit from T[i] */ + *tmpc++ &= MP_MASK; + } + } + + /* add carry */ + *tmpc++ = u; + + /* clear digits above oldused */ + for (i = c->used; i < olduse; i++) { + *tmpc++ = 0; + } + } + + mp_clamp(c); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_S_MP_EXPTMOD_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +int s_mp_exptmod(mp_int *G, mp_int *X, mp_int *P, mp_int *Y, int redmode) { + mp_int M[TAB_SIZE], res, mu; + mp_digit buf; + int err, bitbuf, bitcpy, bitcnt, mode, digidx, x, y, winsize; + + int (*redux)(mp_int *, mp_int *, mp_int *); + + /* find window size */ + x = mp_count_bits(X); + if (x <= 7) { + winsize = 2; + } else if (x <= 36) { + winsize = 3; + } else if (x <= 140) { + winsize = 4; + } else if (x <= 450) { + winsize = 5; + } else if (x <= 1303) { + winsize = 6; + } else if (x <= 3529) { + winsize = 7; + } else { + winsize = 8; + } + + #ifdef MP_LOW_MEM + if (winsize > 5) { + winsize = 5; + } + #endif + + /* init M array */ + /* init first cell */ + if ((err = mp_init(&M[1])) != MP_OKAY) { + return err; + } + + /* now init the second half of the array */ + for (x = 1 << (winsize - 1); x < (1 << winsize); x++) { + if ((err = mp_init(&M[x])) != MP_OKAY) { + for (y = 1 << (winsize - 1); y < x; y++) { + mp_clear(&M[y]); + } + mp_clear(&M[1]); + return err; + } + } + + /* create mu, used for Barrett reduction */ + if ((err = mp_init(&mu)) != MP_OKAY) { + goto LBL_M; + } + + if (redmode == 0) { + if ((err = mp_reduce_setup(&mu, P)) != MP_OKAY) { + goto LBL_MU; + } + redux = mp_reduce; + } else { + if ((err = mp_reduce_2k_setup_l(P, &mu)) != MP_OKAY) { + goto LBL_MU; + } + redux = mp_reduce_2k_l; + } + + /* create M table + * + * The M table contains powers of the base, + * e.g. M[x] = G**x mod P + * + * The first half of the table is not + * computed though accept for M[0] and M[1] + */ + if ((err = mp_mod(G, P, &M[1])) != MP_OKAY) { + goto LBL_MU; + } + + /* compute the value at M[1<<(winsize-1)] by squaring + * M[1] (winsize-1) times + */ + if ((err = mp_copy(&M[1], &M[1 << (winsize - 1)])) != MP_OKAY) { + goto LBL_MU; + } + + for (x = 0; x < (winsize - 1); x++) { + /* square it */ + if ((err = mp_sqr(&M[1 << (winsize - 1)], + &M[1 << (winsize - 1)])) != MP_OKAY) { + goto LBL_MU; + } + + /* reduce modulo P */ + if ((err = redux(&M[1 << (winsize - 1)], P, &mu)) != MP_OKAY) { + goto LBL_MU; + } + } + + /* create upper table, that is M[x] = M[x-1] * M[1] (mod P) + * for x = (2**(winsize - 1) + 1) to (2**winsize - 1) + */ + for (x = (1 << (winsize - 1)) + 1; x < (1 << winsize); x++) { + if ((err = mp_mul(&M[x - 1], &M[1], &M[x])) != MP_OKAY) { + goto LBL_MU; + } + if ((err = redux(&M[x], P, &mu)) != MP_OKAY) { + goto LBL_MU; + } + } + + /* setup result */ + if ((err = mp_init(&res)) != MP_OKAY) { + goto LBL_MU; + } + mp_set(&res, 1); + + /* set initial mode and bit cnt */ + mode = 0; + bitcnt = 1; + buf = 0; + digidx = X->used - 1; + bitcpy = 0; + bitbuf = 0; + + for ( ; ; ) { + /* grab next digit as required */ + if (--bitcnt == 0) { + /* if digidx == -1 we are out of digits */ + if (digidx == -1) { + break; + } + /* read next digit and reset the bitcnt */ + buf = X->dp[digidx--]; + bitcnt = (int)DIGIT_BIT; + } + + /* grab the next msb from the exponent */ + y = (buf >> (mp_digit)(DIGIT_BIT - 1)) & 1; + buf <<= (mp_digit)1; + + /* if the bit is zero and mode == 0 then we ignore it + * These represent the leading zero bits before the first 1 bit + * in the exponent. Technically this opt is not required but it + * does lower the # of trivial squaring/reductions used + */ + if ((mode == 0) && (y == 0)) { + continue; + } + + /* if the bit is zero and mode == 1 then we square */ + if ((mode == 1) && (y == 0)) { + if ((err = mp_sqr(&res, &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, &mu)) != MP_OKAY) { + goto LBL_RES; + } + continue; + } + + /* else we add it to the window */ + bitbuf |= (y << (winsize - ++bitcpy)); + mode = 2; + + if (bitcpy == winsize) { + /* ok window is filled so square as required and multiply */ + /* square first */ + for (x = 0; x < winsize; x++) { + if ((err = mp_sqr(&res, &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, &mu)) != MP_OKAY) { + goto LBL_RES; + } + } + + /* then multiply */ + if ((err = mp_mul(&res, &M[bitbuf], &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, &mu)) != MP_OKAY) { + goto LBL_RES; + } + + /* empty window and reset */ + bitcpy = 0; + bitbuf = 0; + mode = 1; + } + } + + /* if bits remain then square/multiply */ + if ((mode == 2) && (bitcpy > 0)) { + /* square then multiply if the bit is set */ + for (x = 0; x < bitcpy; x++) { + if ((err = mp_sqr(&res, &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, &mu)) != MP_OKAY) { + goto LBL_RES; + } + + bitbuf <<= 1; + if ((bitbuf & (1 << winsize)) != 0) { + /* then multiply */ + if ((err = mp_mul(&res, &M[1], &res)) != MP_OKAY) { + goto LBL_RES; + } + if ((err = redux(&res, P, &mu)) != MP_OKAY) { + goto LBL_RES; + } + } + } + } + + mp_exch(&res, Y); + err = MP_OKAY; +LBL_RES: mp_clear(&res); +LBL_MU: mp_clear(&mu); +LBL_M: + mp_clear(&M[1]); + for (x = 1 << (winsize - 1); x < (1 << winsize); x++) { + mp_clear(&M[x]); + } + return err; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_S_MP_MUL_DIGS_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* multiplies |a| * |b| and only computes upto digs digits of result + * HAC pp. 595, Algorithm 14.12 Modified so you can control how + * many digits of output are created. + */ +int s_mp_mul_digs(mp_int *a, mp_int *b, mp_int *c, int digs) { + mp_int t; + int res, pa, pb, ix, iy; + mp_digit u; + mp_word r; + mp_digit tmpx, *tmpt, *tmpy; + + /* can we use the fast multiplier? */ + if (((digs) < MP_WARRAY) && + (MIN(a->used, b->used) < + (1 << ((CHAR_BIT * sizeof(mp_word)) - (2 * DIGIT_BIT))))) { + return fast_s_mp_mul_digs(a, b, c, digs); + } + + if ((res = mp_init_size(&t, digs)) != MP_OKAY) { + return res; + } + t.used = digs; + + /* compute the digits of the product directly */ + pa = a->used; + for (ix = 0; ix < pa; ix++) { + /* set the carry to zero */ + u = 0; + + /* limit ourselves to making digs digits of output */ + pb = MIN(b->used, digs - ix); + + /* setup some aliases */ + /* copy of the digit from a used within the nested loop */ + tmpx = a->dp[ix]; + + /* an alias for the destination shifted ix places */ + tmpt = t.dp + ix; + + /* an alias for the digits of b */ + tmpy = b->dp; + + /* compute the columns of the output and propagate the carry */ + for (iy = 0; iy < pb; iy++) { + /* compute the column as a mp_word */ + r = (mp_word) * tmpt + + ((mp_word)tmpx * (mp_word) * tmpy++) + + (mp_word)u; + + /* the new column is the lower part of the result */ + *tmpt++ = (mp_digit)(r & ((mp_word)MP_MASK)); + + /* get the carry word from the result */ + u = (mp_digit)(r >> ((mp_word)DIGIT_BIT)); + } + /* set carry if it is placed below digs */ + if ((ix + iy) < digs) { + *tmpt = u; + } + } + + mp_clamp(&t); + mp_exch(&t, c); + + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_S_MP_MUL_HIGH_DIGS_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* multiplies |a| * |b| and does not compute the lower digs digits + * [meant to get the higher part of the product] + */ +int +s_mp_mul_high_digs(mp_int *a, mp_int *b, mp_int *c, int digs) { + mp_int t; + int res, pa, pb, ix, iy; + mp_digit u; + mp_word r; + mp_digit tmpx, *tmpt, *tmpy; + + /* can we use the fast multiplier? */ + #ifdef BN_FAST_S_MP_MUL_HIGH_DIGS_C + if (((a->used + b->used + 1) < MP_WARRAY) && + (MIN(a->used, b->used) < (1 << ((CHAR_BIT * sizeof(mp_word)) - (2 * DIGIT_BIT))))) { + return fast_s_mp_mul_high_digs(a, b, c, digs); + } + #endif + + if ((res = mp_init_size(&t, a->used + b->used + 1)) != MP_OKAY) { + return res; + } + t.used = a->used + b->used + 1; + + pa = a->used; + pb = b->used; + for (ix = 0; ix < pa; ix++) { + /* clear the carry */ + u = 0; + + /* left hand side of A[ix] * B[iy] */ + tmpx = a->dp[ix]; + + /* alias to the address of where the digits will be stored */ + tmpt = &(t.dp[digs]); + + /* alias for where to read the right hand side from */ + tmpy = b->dp + (digs - ix); + + for (iy = digs - ix; iy < pb; iy++) { + /* calculate the double precision result */ + r = (mp_word) * tmpt + + ((mp_word)tmpx * (mp_word) * tmpy++) + + (mp_word)u; + + /* get the lower part */ + *tmpt++ = (mp_digit)(r & ((mp_word)MP_MASK)); + + /* carry the carry */ + u = (mp_digit)(r >> ((mp_word)DIGIT_BIT)); + } + *tmpt = u; + } + mp_clamp(&t); + mp_exch(&t, c); + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_S_MP_SQR_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* low level squaring, b = a*a, HAC pp.596-597, Algorithm 14.16 */ +int s_mp_sqr(mp_int *a, mp_int *b) { + mp_int t; + int res, ix, iy, pa; + mp_word r; + mp_digit u, tmpx, *tmpt; + + pa = a->used; + if ((res = mp_init_size(&t, (2 * pa) + 1)) != MP_OKAY) { + return res; + } + + /* default used is maximum possible size */ + t.used = (2 * pa) + 1; + + for (ix = 0; ix < pa; ix++) { + /* first calculate the digit at 2*ix */ + /* calculate double precision result */ + r = (mp_word)t.dp[2 * ix] + + ((mp_word)a->dp[ix] * (mp_word)a->dp[ix]); + + /* store lower part in result */ + t.dp[ix + ix] = (mp_digit)(r & ((mp_word)MP_MASK)); + + /* get the carry */ + u = (mp_digit)(r >> ((mp_word)DIGIT_BIT)); + + /* left hand side of A[ix] * A[iy] */ + tmpx = a->dp[ix]; + + /* alias for where to store the results */ + tmpt = t.dp + ((2 * ix) + 1); + + for (iy = ix + 1; iy < pa; iy++) { + /* first calculate the product */ + r = ((mp_word)tmpx) * ((mp_word)a->dp[iy]); + + /* now calculate the double precision result, note we use + * addition instead of *2 since it's easier to optimize + */ + r = ((mp_word) * tmpt) + r + r + ((mp_word)u); + + /* store lower part */ + *tmpt++ = (mp_digit)(r & ((mp_word)MP_MASK)); + + /* get carry */ + u = (mp_digit)(r >> ((mp_word)DIGIT_BIT)); + } + /* propagate upwards */ + while (u != ((mp_digit)0)) { + r = ((mp_word) * tmpt) + ((mp_word)u); + *tmpt++ = (mp_digit)(r & ((mp_word)MP_MASK)); + u = (mp_digit)(r >> ((mp_word)DIGIT_BIT)); + } + } + + mp_clamp(&t); + mp_exch(&t, b); + mp_clear(&t); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BN_S_MP_SUB_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* low level subtraction (assumes |a| > |b|), HAC pp.595 Algorithm 14.9 */ +int +s_mp_sub(mp_int *a, mp_int *b, mp_int *c) { + int olduse, res, min, max; + + /* find sizes */ + min = b->used; + max = a->used; + + /* init result */ + if (c->alloc < max) { + if ((res = mp_grow(c, max)) != MP_OKAY) { + return res; + } + } + olduse = c->used; + c->used = max; + + { + mp_digit u, *tmpa, *tmpb, *tmpc; + int i; + + /* alias for digit pointers */ + tmpa = a->dp; + tmpb = b->dp; + tmpc = c->dp; + + /* set carry to zero */ + u = 0; + for (i = 0; i < min; i++) { + /* T[i] = A[i] - B[i] - U */ + *tmpc = (*tmpa++ - *tmpb++) - u; + + /* U = carry bit of T[i] + * Note this saves performing an AND operation since + * if a carry does occur it will propagate all the way to the + * MSB. As a result a single shift is enough to get the carry + */ + u = *tmpc >> ((mp_digit)((CHAR_BIT * sizeof(mp_digit)) - 1)); + + /* Clear carry from T[i] */ + *tmpc++ &= MP_MASK; + } + + /* now copy higher words if any, e.g. if A has more digits than B */ + for ( ; i < max; i++) { + /* T[i] = A[i] - U */ + *tmpc = *tmpa++ - u; + + /* U = carry bit of T[i] */ + u = *tmpc >> ((mp_digit)((CHAR_BIT * sizeof(mp_digit)) - 1)); + + /* Clear carry from T[i] */ + *tmpc++ &= MP_MASK; + } + + /* clear digits above used (since we may not have grown result above) */ + for (i = c->used; i < olduse; i++) { + *tmpc++ = 0; + } + } + + mp_clamp(c); + return MP_OKAY; +} +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + + +#ifdef BNCORE_C + +/* LibTomMath, multiple-precision integer library -- Tom St Denis + * + * LibTomMath is a library that provides multiple-precision + * integer arithmetic as well as number theoretic functionality. + * + * The library was designed directly after the MPI library by + * Michael Fromberger but has been written from scratch with + * additional optimizations in place. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tstdenis82@gmail.com, http://libtom.org + */ + +/* Known optimal configurations + + CPU /Compiler /MUL CUTOFF/SQR CUTOFF + ------------------------------------------------------------- + Intel P4 Northwood /GCC v3.4.1 / 88/ 128/LTM 0.32 ;-) + AMD Athlon64 /GCC v3.4.4 / 80/ 120/LTM 0.35 + + */ + +int KARATSUBA_MUL_CUTOFF = 80, /* Min. number of digits before Karatsuba multiplication is used. */ + KARATSUBA_SQR_CUTOFF = 120, /* Min. number of digits before Karatsuba squaring is used. */ + + TOOM_MUL_CUTOFF = 350, /* no optimal values of these are known yet so set em high */ + TOOM_SQR_CUTOFF = 400; +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file crypt.c + Build strings, Tom St Denis + */ + +const char *crypt_build_settings = + "LibTomCrypt ""1.17"" (Tom St Denis, tomstdenis@gmail.com)\n" + "LibTomCrypt is public domain software.\n" + "Built on " __DATE__ " at " __TIME__ "\n\n\n" + "Endianess: " +#if defined(ENDIAN_NEUTRAL) + "neutral\n" +#elif defined(ENDIAN_LITTLE) + "little" + #if defined(ENDIAN_32BITWORD) + " (32-bit words)\n" + #else + " (64-bit words)\n" + #endif +#elif defined(ENDIAN_BIG) + "big" + #if defined(ENDIAN_32BITWORD) + " (32-bit words)\n" + #else + " (64-bit words)\n" + #endif +#endif + "Clean stack: " +#if defined(LTC_CLEAN_STACK) + "enabled\n" +#else + "disabled\n" +#endif + "Ciphers built-in:\n" +#if defined(LTC_BLOWFISH) + " Blowfish\n" +#endif +#if defined(LTC_RC2) + " LTC_RC2\n" +#endif +#if defined(LTC_RC5) + " LTC_RC5\n" +#endif +#if defined(LTC_RC6) + " LTC_RC6\n" +#endif +#if defined(LTC_SAFERP) + " Safer+\n" +#endif +#if defined(LTC_SAFER) + " Safer\n" +#endif +#if defined(LTC_RIJNDAEL) + " Rijndael\n" +#endif +#if defined(LTC_XTEA) + " LTC_XTEA\n" +#endif +#if defined(LTC_TWOFISH) + " Twofish " + #if defined(LTC_TWOFISH_SMALL) && defined(LTC_TWOFISH_TABLES) && defined(LTC_TWOFISH_ALL_TABLES) + "(small, tables, all_tables)\n" + #elif defined(LTC_TWOFISH_SMALL) && defined(LTC_TWOFISH_TABLES) + "(small, tables)\n" + #elif defined(LTC_TWOFISH_SMALL) && defined(LTC_TWOFISH_ALL_TABLES) + "(small, all_tables)\n" + #elif defined(LTC_TWOFISH_TABLES) && defined(LTC_TWOFISH_ALL_TABLES) + "(tables, all_tables)\n" + #elif defined(LTC_TWOFISH_SMALL) + "(small)\n" + #elif defined(LTC_TWOFISH_TABLES) + "(tables)\n" + #elif defined(LTC_TWOFISH_ALL_TABLES) + "(all_tables)\n" + #else + "\n" + #endif +#endif +#if defined(LTC_DES) + " LTC_DES\n" +#endif +#if defined(LTC_CAST5) + " LTC_CAST5\n" +#endif +#if defined(LTC_NOEKEON) + " Noekeon\n" +#endif +#if defined(LTC_SKIPJACK) + " Skipjack\n" +#endif +#if defined(LTC_KHAZAD) + " Khazad\n" +#endif +#if defined(LTC_ANUBIS) + " Anubis " +#endif +#if defined(LTC_ANUBIS_TWEAK) + " (tweaked)" +#endif + "\n" +#if defined(LTC_KSEED) + " LTC_KSEED\n" +#endif +#if defined(LTC_KASUMI) + " KASUMI\n" +#endif + + "\nHashes built-in:\n" +#if defined(LTC_SHA512) + " LTC_SHA-512\n" +#endif +#if defined(LTC_SHA384) + " LTC_SHA-384\n" +#endif +#if defined(LTC_SHA256) + " LTC_SHA-256\n" +#endif +#if defined(LTC_SHA224) + " LTC_SHA-224\n" +#endif +#if defined(LTC_TIGER) + " LTC_TIGER\n" +#endif +#if defined(LTC_SHA1) + " LTC_SHA1\n" +#endif +#if defined(LTC_MD5) + " LTC_MD5\n" +#endif +#if defined(LTC_MD4) + " LTC_MD4\n" +#endif +#if defined(LTC_MD2) + " LTC_MD2\n" +#endif +#if defined(LTC_RIPEMD128) + " LTC_RIPEMD128\n" +#endif +#if defined(LTC_RIPEMD160) + " LTC_RIPEMD160\n" +#endif +#if defined(LTC_RIPEMD256) + " LTC_RIPEMD256\n" +#endif +#if defined(LTC_RIPEMD320) + " LTC_RIPEMD320\n" +#endif +#if defined(LTC_WHIRLPOOL) + " LTC_WHIRLPOOL\n" +#endif +#if defined(LTC_CHC_HASH) + " LTC_CHC_HASH \n" +#endif + + "\nBlock Chaining Modes:\n" +#if defined(LTC_CFB_MODE) + " CFB\n" +#endif +#if defined(LTC_OFB_MODE) + " OFB\n" +#endif +#if defined(LTC_ECB_MODE) + " ECB\n" +#endif +#if defined(LTC_CBC_MODE) + " CBC\n" +#endif +#if defined(LTC_CTR_MODE) + " CTR " +#endif +#if defined(LTC_CTR_OLD) + " (CTR_OLD) " +#endif + "\n" +#if defined(LRW_MODE) + " LRW_MODE" + #if defined(LRW_TABLES) + " (LRW_TABLES) " + #endif + "\n" +#endif +#if defined(LTC_F8_MODE) + " F8 MODE\n" +#endif +#if defined(LTC_XTS_MODE) + " LTC_XTS_MODE\n" +#endif + + "\nMACs:\n" +#if defined(LTC_HMAC) + " LTC_HMAC\n" +#endif +#if defined(LTC_OMAC) + " LTC_OMAC\n" +#endif +#if defined(LTC_PMAC) + " PMAC\n" +#endif +#if defined(LTC_PELICAN) + " LTC_PELICAN\n" +#endif +#if defined(LTC_XCBC) + " XCBC-MAC\n" +#endif +#if defined(LTC_F9_MODE) + " F9-MAC\n" +#endif + + "\nENC + AUTH modes:\n" +#if defined(LTC_EAX_MODE) + " LTC_EAX_MODE\n" +#endif +#if defined(LTC_OCB_MODE) + " LTC_OCB_MODE\n" +#endif +#if defined(LTC_CCM_MODE) + " LTC_CCM_MODE\n" +#endif +#if defined(LTC_GCM_MODE) + " LTC_GCM_MODE " +#endif +#if defined(LTC_GCM_TABLES) + " (LTC_GCM_TABLES) " +#endif + "\n" + + "\nPRNG:\n" +#if defined(LTC_YARROW) + " Yarrow\n" +#endif +#if defined(LTC_SPRNG) + " LTC_SPRNG\n" +#endif +#if defined(LTC_RC4) + " LTC_RC4\n" +#endif +#if defined(LTC_FORTUNA) + " Fortuna\n" +#endif +#if defined(LTC_SOBER128) + " LTC_SOBER128\n" +#endif + + "\nPK Algs:\n" +#if defined(LTC_MRSA) + " RSA \n" +#endif +#if defined(LTC_MECC) + " ECC\n" +#endif +#if defined(LTC_MDSA) + " DSA\n" +#endif +#if defined(MKAT) + " Katja\n" +#endif + + "\nCompiler:\n" +#if defined(WIN32) + " WIN32 platform detected.\n" +#endif +#if defined(__CYGWIN__) + " CYGWIN Detected.\n" +#endif +#if defined(__DJGPP__) + " DJGPP Detected.\n" +#endif +#if defined(_MSC_VER) + " MSVC compiler detected.\n" +#endif +#if defined(__GNUC__) + " GCC compiler detected.\n" +#endif +#if defined(INTEL_CC) + " Intel C Compiler detected.\n" +#endif +#if defined(__x86_64__) + " x86-64 detected.\n" +#endif +#if defined(LTC_PPC32) + " LTC_PPC32 defined \n" +#endif + + "\nVarious others: " +#if defined(LTC_BASE64) + " LTC_BASE64 " +#endif +#if defined(MPI) + " MPI " +#endif +#if defined(TRY_UNRANDOM_FIRST) + " TRY_UNRANDOM_FIRST " +#endif +#if defined(LTC_TEST) + " LTC_TEST " +#endif +#if defined(LTC_PKCS_1) + " LTC_PKCS#1 " +#endif +#if defined(LTC_PKCS_5) + " LTC_PKCS#5 " +#endif +#if defined(LTC_SMALL_CODE) + " LTC_SMALL_CODE " +#endif +#if defined(LTC_NO_FILE) + " LTC_NO_FILE " +#endif +#if defined(LTC_DER) + " LTC_DER " +#endif +#if defined(LTC_FAST) + " LTC_FAST " +#endif +#if defined(LTC_NO_FAST) + " LTC_NO_FAST " +#endif +#if defined(LTC_NO_BSWAP) + " LTC_NO_BSWAP " +#endif +#if defined(LTC_NO_ASM) + " LTC_NO_ASM " +#endif +#if defined(LTC_NO_TEST) + " LTC_NO_TEST " +#endif +#if defined(LTC_NO_TABLES) + " LTC_NO_TABLES " +#endif +#if defined(LTC_PTHREAD) + " LTC_PTHREAD " +#endif +#if defined(LTM_LTC_DESC) + " LTM_DESC " +#endif +#if defined(TFM_LTC_DESC) + " TFM_DESC " +#endif +#if defined(LTC_MECC_ACCEL) + " LTC_MECC_ACCEL " +#endif +#if defined(GMP_LTC_DESC) + " GMP_DESC " +#endif +#if defined(LTC_EASY) + " (easy) " +#endif +#if defined(LTC_MECC_FP) + " LTC_MECC_FP " +#endif +#if defined(LTC_ECC_SHAMIR) + " LTC_ECC_SHAMIR " +#endif + "\n" + "\n\n\n" +; + + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt.c,v $ */ +/* $Revision: 1.36 $ */ +/* $Date: 2007/05/12 14:46:12 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ +#include + +/** + @file crypt_argchk.c + Perform argument checking, Tom St Denis + */ + +#if (ARGTYPE == 0) +void crypt_argchk(char *v, char *s, int d) { + fprintf(stderr, "LTC_ARGCHK '%s' failure on line %d of file %s\n", + v, d, s); + (void)raise(SIGABRT); +} +#endif + +#ifndef TOMCRYPT_H_ +#define TOMCRYPT_H_ +#define USE_LTM +#define LTM_DESC +#define LTC_SHA1 +#include +#include +#include +#include +#include +#include +#include + +/* use configuration data */ +#ifndef TOMCRYPT_CUSTOM_H_ +#define TOMCRYPT_CUSTOM_H_ + +/* macros for various libc functions you can change for embedded targets */ +#ifndef XMALLOC + #ifdef malloc + #define LTC_NO_PROTOTYPES + #endif + #define XMALLOC malloc +#endif +#ifndef XREALLOC + #ifdef realloc + #define LTC_NO_PROTOTYPES + #endif + #define XREALLOC realloc +#endif +#ifndef XCALLOC + #ifdef calloc + #define LTC_NO_PROTOTYPES + #endif + #define XCALLOC calloc +#endif +#ifndef XFREE + #ifdef free + #define LTC_NO_PROTOTYPES + #endif + #define XFREE free +#endif + +#ifndef XMEMSET + #ifdef memset + #define LTC_NO_PROTOTYPES + #endif + #define XMEMSET memset +#endif +#ifndef XMEMCPY + #ifdef memcpy + #define LTC_NO_PROTOTYPES + #endif + #define XMEMCPY memcpy +#endif +#ifndef XMEMCMP + #ifdef memcmp + #define LTC_NO_PROTOTYPES + #endif + #define XMEMCMP memcmp +#endif +#ifndef XSTRCMP + #ifdef strcmp + #define LTC_NO_PROTOTYPES + #endif + #define XSTRCMP strcmp +#endif + +#ifndef XCLOCK + #define XCLOCK clock +#endif +#ifndef XCLOCKS_PER_SEC + #define XCLOCKS_PER_SEC CLOCKS_PER_SEC +#endif + +#ifndef XQSORT + #ifdef qsort + #define LTC_NO_PROTOTYPES + #endif + #define XQSORT qsort +#endif + +/* Easy button? */ +#ifdef LTC_EASY + #define LTC_NO_CIPHERS + #define LTC_RIJNDAEL + #define LTC_BLOWFISH + #define LTC_DES + #define LTC_CAST5 + + #define LTC_NO_MODES + #define LTC_ECB_MODE + #define LTC_CBC_MODE + #define LTC_CTR_MODE + + #define LTC_NO_HASHES + #define LTC_SHA1 + #define LTC_SHA512 + #define LTC_SHA384 + #define LTC_SHA256 + #define LTC_SHA224 + + #define LTC_NO_MACS + #define LTC_HMAC + #define LTC_OMAC + #define LTC_CCM_MODE + + #define LTC_NO_PRNGS + #define LTC_SPRNG + #define LTC_YARROW + #define LTC_DEVRANDOM + #define TRY_URANDOM_FIRST + + #define LTC_NO_PK + #define LTC_MRSA + #define LTC_MECC +#endif + +/* Use small code where possible */ +/* #define LTC_SMALL_CODE */ + +/* Enable self-test test vector checking */ +#ifndef LTC_NO_TEST + #define LTC_TEST +#endif + +/* clean the stack of functions which put private information on stack */ +/* #define LTC_CLEAN_STACK */ + +/* disable all file related functions */ +/* #define LTC_NO_FILE */ + +/* disable all forms of ASM */ +/* #define LTC_NO_ASM */ + +/* disable FAST mode */ +/* #define LTC_NO_FAST */ + +/* disable BSWAP on x86 */ +/* #define LTC_NO_BSWAP */ + +/* ---> Symmetric Block Ciphers <--- */ +#ifndef LTC_NO_CIPHERS + + #define LTC_BLOWFISH + #define LTC_RC2 + #define LTC_RC5 + #define LTC_RC6 + #define LTC_SAFERP + #define LTC_RIJNDAEL + #define LTC_XTEA + +/* _TABLES tells it to use tables during setup, _SMALL means to use the smaller scheduled key format + * (saves 4KB of ram), _ALL_TABLES enables all tables during setup */ + #define LTC_TWOFISH + #ifndef LTC_NO_TABLES + #define LTC_TWOFISH_TABLES +/* #define LTC_TWOFISH_ALL_TABLES */ + #else + #define LTC_TWOFISH_SMALL + #endif +/* #define LTC_TWOFISH_SMALL */ +/* LTC_DES includes EDE triple-LTC_DES */ + #define LTC_DES + #define LTC_CAST5 + #define LTC_NOEKEON + #define LTC_SKIPJACK + #define LTC_SAFER + #define LTC_KHAZAD + #define LTC_ANUBIS + #define LTC_ANUBIS_TWEAK + #define LTC_KSEED + #define LTC_KASUMI +#endif /* LTC_NO_CIPHERS */ + + +/* ---> Block Cipher Modes of Operation <--- */ +#ifndef LTC_NO_MODES + + #define LTC_CFB_MODE + #define LTC_OFB_MODE + #define LTC_ECB_MODE + #define LTC_CBC_MODE + #define LTC_CTR_MODE + +/* F8 chaining mode */ + #define LTC_F8_MODE + +/* LRW mode */ + #define LTC_LRW_MODE + #ifndef LTC_NO_TABLES + +/* like GCM mode this will enable 16 8x128 tables [64KB] that make + * seeking very fast. + */ + #define LRW_TABLES + #endif + +/* XTS mode */ + #define LTC_XTS_MODE +#endif /* LTC_NO_MODES */ + +/* ---> One-Way Hash Functions <--- */ +#ifndef LTC_NO_HASHES + + #define LTC_CHC_HASH + #define LTC_WHIRLPOOL + #define LTC_SHA512 + #define LTC_SHA384 + #define LTC_SHA256 + #define LTC_SHA224 + #define LTC_TIGER + #define LTC_SHA1 + #define LTC_MD5 + #define LTC_MD4 + #define LTC_MD2 + #define LTC_RIPEMD128 + #define LTC_RIPEMD160 + #define LTC_RIPEMD256 + #define LTC_RIPEMD320 +#endif /* LTC_NO_HASHES */ + +/* ---> MAC functions <--- */ +#ifndef LTC_NO_MACS + + #define LTC_HMAC + #define LTC_OMAC + #define LTC_PMAC + #define LTC_XCBC + #define LTC_F9_MODE + #define LTC_PELICAN + + #if defined(LTC_PELICAN) && !defined(LTC_RIJNDAEL) + #error Pelican-MAC requires LTC_RIJNDAEL + #endif + +/* ---> Encrypt + Authenticate Modes <--- */ + + #define LTC_EAX_MODE + #if defined(LTC_EAX_MODE) && !(defined(LTC_CTR_MODE) && defined(LTC_OMAC)) + #error LTC_EAX_MODE requires CTR and LTC_OMAC mode + #endif + + #define LTC_OCB_MODE + #define LTC_CCM_MODE + #define LTC_GCM_MODE + +/* Use 64KiB tables */ + #ifndef LTC_NO_TABLES + #define LTC_GCM_TABLES + #endif + +/* USE SSE2? requires GCC works on x86_32 and x86_64*/ + #ifdef LTC_GCM_TABLES +/* #define LTC_GCM_TABLES_SSE2 */ + #endif +#endif /* LTC_NO_MACS */ + +/* Various tidbits of modern neatoness */ +#define LTC_BASE64 + +/* --> Pseudo Random Number Generators <--- */ +#ifndef LTC_NO_PRNGS + +/* Yarrow */ + #define LTC_YARROW +/* which descriptor of AES to use? */ +/* 0 = rijndael_enc 1 = aes_enc, 2 = rijndael [full], 3 = aes [full] */ + #define LTC_YARROW_AES 0 + + #if defined(LTC_YARROW) && !defined(LTC_CTR_MODE) + #error LTC_YARROW requires LTC_CTR_MODE chaining mode to be defined! + #endif + +/* a PRNG that simply reads from an available system source */ + #define LTC_SPRNG + +/* The LTC_RC4 stream cipher */ + #define LTC_RC4 + +/* Fortuna PRNG */ + #define LTC_FORTUNA +/* reseed every N calls to the read function */ + #define LTC_FORTUNA_WD 10 +/* number of pools (4..32) can save a bit of ram by lowering the count */ + #define LTC_FORTUNA_POOLS 32 + +/* Greg's LTC_SOBER128 PRNG ;-0 */ + #define LTC_SOBER128 + +/* the *nix style /dev/random device */ + #define LTC_DEVRANDOM +/* try /dev/urandom before trying /dev/random */ + #define TRY_URANDOM_FIRST +#endif /* LTC_NO_PRNGS */ + +/* ---> math provider? <--- */ +#ifndef LTC_NO_MATH + +/* LibTomMath */ +/* #define LTM_LTC_DESC */ + +/* TomsFastMath */ +/* #define TFM_LTC_DESC */ +#endif /* LTC_NO_MATH */ + +/* ---> Public Key Crypto <--- */ +#ifndef LTC_NO_PK + +/* Include RSA support */ + #define LTC_MRSA + +/* Include Katja (a Rabin variant like RSA) */ +/* #define MKAT */ + +/* Digital Signature Algorithm */ + #define LTC_MDSA + +/* ECC */ + #define LTC_MECC + +/* use Shamir's trick for point mul (speeds up signature verification) */ + #define LTC_ECC_SHAMIR + + #if defined(TFM_LTC_DESC) && defined(LTC_MECC) + #define LTC_MECC_ACCEL + #endif + +/* do we want fixed point ECC */ +/* #define LTC_MECC_FP */ + +/* Timing Resistant? */ +/* #define LTC_ECC_TIMING_RESISTANT */ +#endif /* LTC_NO_PK */ + +/* LTC_PKCS #1 (RSA) and #5 (Password Handling) stuff */ +#ifndef LTC_NO_PKCS + + #define LTC_PKCS_1 + #define LTC_PKCS_5 + +/* Include ASN.1 DER (required by DSA/RSA) */ + #define LTC_DER +#endif /* LTC_NO_PKCS */ + +/* cleanup */ + +#ifdef LTC_MECC +/* Supported ECC Key Sizes */ + #ifndef LTC_NO_CURVES + #define ECC112 + #define ECC128 + #define ECC160 + #define ECC192 + #define ECC224 + #define ECC256 + #define ECC384 + #define ECC521 + #endif +#endif + +#if defined(LTC_MECC) || defined(LTC_MRSA) || defined(LTC_MDSA) || defined(MKATJA) +/* Include the MPI functionality? (required by the PK algorithms) */ + #define MPI +#endif + +#ifdef LTC_MRSA + #define LTC_PKCS_1 +#endif + +#if defined(LTC_DER) && !defined(MPI) + #error ASN.1 DER requires MPI functionality +#endif + +#if (defined(LTC_MDSA) || defined(LTC_MRSA) || defined(LTC_MECC) || defined(MKATJA)) && !defined(LTC_DER) + #error PK requires ASN.1 DER functionality, make sure LTC_DER is enabled +#endif + +/* THREAD management */ +#ifdef LTC_PTHREAD + + #include + + #define LTC_MUTEX_GLOBAL(x) pthread_mutex_t x = PTHREAD_MUTEX_INITIALIZER; + #define LTC_MUTEX_PROTO(x) extern pthread_mutex_t x; + #define LTC_MUTEX_TYPE(x) pthread_mutex_t x; + #define LTC_MUTEX_INIT(x) pthread_mutex_init(x, NULL); + #define LTC_MUTEX_LOCK(x) pthread_mutex_lock(x); + #define LTC_MUTEX_UNLOCK(x) pthread_mutex_unlock(x); + +#else + +/* default no functions */ + #define LTC_MUTEX_GLOBAL(x) + #define LTC_MUTEX_PROTO(x) + #define LTC_MUTEX_TYPE(x) + #define LTC_MUTEX_INIT(x) + #define LTC_MUTEX_LOCK(x) + #define LTC_MUTEX_UNLOCK(x) +#endif + +/* Debuggers */ + +/* define this if you use Valgrind, note: it CHANGES the way SOBER-128 and LTC_RC4 work (see the code) */ +/* #define LTC_VALGRIND */ +#endif + + + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_custom.h,v $ */ +/* $Revision: 1.73 $ */ +/* $Date: 2007/05/12 14:37:41 $ */ + +#ifdef __cplusplus +extern "C" { +#endif + +/* version */ +#define CRYPT 0x0117 +#define SCRYPT "1.17" + +/* max size of either a cipher/hash block or symmetric key [largest of the two] */ +#define MAXBLOCKSIZE 128 + +/* descriptor table size */ + +/* error codes [will be expanded in future releases] */ +enum { + CRYPT_OK=0, /* Result OK */ + CRYPT_ERROR, /* Generic Error */ + CRYPT_NOP, /* Not a failure but no operation was performed */ + + CRYPT_INVALID_KEYSIZE, /* Invalid key size given */ + CRYPT_INVALID_ROUNDS, /* Invalid number of rounds */ + CRYPT_FAIL_TESTVECTOR, /* Algorithm failed test vectors */ + + CRYPT_BUFFER_OVERFLOW, /* Not enough space for output */ + CRYPT_INVALID_PACKET, /* Invalid input packet given */ + + CRYPT_INVALID_PRNGSIZE, /* Invalid number of bits for a PRNG */ + CRYPT_ERROR_READPRNG, /* Could not read enough from PRNG */ + + CRYPT_INVALID_CIPHER, /* Invalid cipher specified */ + CRYPT_INVALID_HASH, /* Invalid hash specified */ + CRYPT_INVALID_PRNG, /* Invalid PRNG specified */ + + CRYPT_MEM, /* Out of memory */ + + CRYPT_PK_TYPE_MISMATCH, /* Not equivalent types of PK keys */ + CRYPT_PK_NOT_PRIVATE, /* Requires a private PK key */ + + CRYPT_INVALID_ARG, /* Generic invalid argument */ + CRYPT_FILE_NOTFOUND, /* File Not Found */ + + CRYPT_PK_INVALID_TYPE, /* Invalid type of PK key */ + CRYPT_PK_INVALID_SYSTEM, /* Invalid PK system specified */ + CRYPT_PK_DUP, /* Duplicate key already in key ring */ + CRYPT_PK_NOT_FOUND, /* Key not found in keyring */ + CRYPT_PK_INVALID_SIZE, /* Invalid size input for PK parameters */ + + CRYPT_INVALID_PRIME_SIZE, /* Invalid size of prime requested */ + CRYPT_PK_INVALID_PADDING /* Invalid padding on input */ +}; + +/* This is the build config file. + * + * With this you can setup what to inlcude/exclude automatically during any build. Just comment + * out the line that #define's the word for the thing you want to remove. phew! + */ + +#ifndef TOMCRYPT_CFG_H +#define TOMCRYPT_CFG_H + +#if defined(_WIN32) || defined(_MSC_VER) + #define LTC_CALL __cdecl +#else + #ifndef LTC_CALL + #define LTC_CALL + #endif +#endif + +#ifndef LTC_EXPORT + #define LTC_EXPORT +#endif + +/* certain platforms use macros for these, making the prototypes broken */ +#ifndef LTC_NO_PROTOTYPES + +/* you can change how memory allocation works ... */ +LTC_EXPORT void *LTC_CALL XMALLOC(size_t n); +LTC_EXPORT void *LTC_CALL XREALLOC(void *p, size_t n); +LTC_EXPORT void *LTC_CALL XCALLOC(size_t n, size_t s); +LTC_EXPORT void LTC_CALL XFREE(void *p); + +LTC_EXPORT void LTC_CALL XQSORT(void *base, size_t nmemb, size_t size, int (*compar)(const void *, const void *)); + + +/* change the clock function too */ +LTC_EXPORT clock_t LTC_CALL XCLOCK(void); + +/* various other functions */ +LTC_EXPORT void *LTC_CALL XMEMCPY(void *dest, const void *src, size_t n); +LTC_EXPORT int LTC_CALL XMEMCMP(const void *s1, const void *s2, size_t n); +LTC_EXPORT void *LTC_CALL XMEMSET(void *s, int c, size_t n); + +LTC_EXPORT int LTC_CALL XSTRCMP(const char *s1, const char *s2); +#endif + +/* type of argument checking, 0=default, 1=fatal and 2=error+continue, 3=nothing */ +#ifndef ARGTYPE + #define ARGTYPE 0 +#endif + +/* Controls endianess and size of registers. Leave uncommented to get platform neutral [slower] code + * + * Note: in order to use the optimized macros your platform must support unaligned 32 and 64 bit read/writes. + * The x86 platforms allow this but some others [ARM for instance] do not. On those platforms you **MUST** + * use the portable [slower] macros. + */ + +/* detect x86-32 machines somewhat */ +#if !defined(__STRICT_ANSI__) && (defined(INTEL_CC) || (defined(_MSC_VER) && defined(WIN32)) || (defined(__GNUC__) && (defined(__DJGPP__) || defined(__CYGWIN__) || defined(__MINGW32__) || defined(__i386__)))) + #define ENDIAN_LITTLE + #define ENDIAN_32BITWORD + #define LTC_FAST + #define LTC_FAST_TYPE unsigned long +#endif + +/* detects MIPS R5900 processors (PS2) */ +#if (defined(__R5900) || defined(R5900) || defined(__R5900__)) && (defined(_mips) || defined(__mips__) || defined(mips)) + #define ENDIAN_LITTLE + #define ENDIAN_64BITWORD +#endif + +/* detect amd64 */ +#if !defined(__STRICT_ANSI__) && defined(__x86_64__) + #define ENDIAN_LITTLE + #define ENDIAN_64BITWORD + #define LTC_FAST + #define LTC_FAST_TYPE unsigned long +#endif + +/* detect PPC32 */ +#if !defined(__STRICT_ANSI__) && defined(LTC_PPC32) + #define ENDIAN_BIG + #define ENDIAN_32BITWORD + #define LTC_FAST + #define LTC_FAST_TYPE unsigned long +#endif + +/* detect sparc and sparc64 */ +#if defined(__sparc__) + #define ENDIAN_BIG + #if defined(__arch64__) + #define ENDIAN_64BITWORD + #else + #define ENDIAN_32BITWORD + #endif +#endif + + +#ifdef LTC_NO_FAST + #ifdef LTC_FAST + #undef LTC_FAST + #endif +#endif + +/* No asm is a quick way to disable anything "not portable" */ +#ifdef LTC_NO_ASM + #undef ENDIAN_LITTLE + #undef ENDIAN_BIG + #undef ENDIAN_32BITWORD + #undef ENDIAN_64BITWORD + #undef LTC_FAST + #undef LTC_FAST_TYPE + #define LTC_NO_ROLC + #define LTC_NO_BSWAP +#endif + +/* #define ENDIAN_LITTLE */ +/* #define ENDIAN_BIG */ + +/* #define ENDIAN_32BITWORD */ +/* #define ENDIAN_64BITWORD */ + +#if (defined(ENDIAN_BIG) || defined(ENDIAN_LITTLE)) && !(defined(ENDIAN_32BITWORD) || defined(ENDIAN_64BITWORD)) + #error You must specify a word size as well as endianess in tomcrypt_cfg.h +#endif + +#if !(defined(ENDIAN_BIG) || defined(ENDIAN_LITTLE)) + #define ENDIAN_NEUTRAL +#endif +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_cfg.h,v $ */ +/* $Revision: 1.19 $ */ +/* $Date: 2006/12/04 02:19:48 $ */ + +/* fix for MSVC ...evil! */ +#ifdef _MSC_VER + #define CONST64(n) n ## ui64 +typedef unsigned __int64 ulong64; +#else + #define CONST64(n) n ## ULL +typedef unsigned long long ulong64; +#endif + +/* this is the "32-bit at least" data type + * Re-define it to suit your platform but it must be at least 32-bits + */ +#if defined(__x86_64__) || (defined(__sparc__) && defined(__arch64__)) +typedef unsigned ulong32; +#else +typedef unsigned long ulong32; +#endif + +/* ---- HELPER MACROS ---- */ +#ifdef ENDIAN_NEUTRAL + + #define STORE32L(x, y) \ + { (y)[3] = (unsigned char)(((x) >> 24) & 255); (y)[2] = (unsigned char)(((x) >> 16) & 255); \ + (y)[1] = (unsigned char)(((x) >> 8) & 255); (y)[0] = (unsigned char)((x) & 255); } + + #define LOAD32L(x, y) \ + { x = ((unsigned long)((y)[3] & 255) << 24) | \ + ((unsigned long)((y)[2] & 255) << 16) | \ + ((unsigned long)((y)[1] & 255) << 8) | \ + ((unsigned long)((y)[0] & 255)); } + + #define STORE64L(x, y) \ + { (y)[7] = (unsigned char)(((x) >> 56) & 255); (y)[6] = (unsigned char)(((x) >> 48) & 255); \ + (y)[5] = (unsigned char)(((x) >> 40) & 255); (y)[4] = (unsigned char)(((x) >> 32) & 255); \ + (y)[3] = (unsigned char)(((x) >> 24) & 255); (y)[2] = (unsigned char)(((x) >> 16) & 255); \ + (y)[1] = (unsigned char)(((x) >> 8) & 255); (y)[0] = (unsigned char)((x) & 255); } + + #define LOAD64L(x, y) \ + { x = (((ulong64)((y)[7] & 255)) << 56) | (((ulong64)((y)[6] & 255)) << 48) | \ + (((ulong64)((y)[5] & 255)) << 40) | (((ulong64)((y)[4] & 255)) << 32) | \ + (((ulong64)((y)[3] & 255)) << 24) | (((ulong64)((y)[2] & 255)) << 16) | \ + (((ulong64)((y)[1] & 255)) << 8) | (((ulong64)((y)[0] & 255))); } + + #define STORE32H(x, y) \ + { (y)[0] = (unsigned char)(((x) >> 24) & 255); (y)[1] = (unsigned char)(((x) >> 16) & 255); \ + (y)[2] = (unsigned char)(((x) >> 8) & 255); (y)[3] = (unsigned char)((x) & 255); } + + #define LOAD32H(x, y) \ + { x = ((unsigned long)((y)[0] & 255) << 24) | \ + ((unsigned long)((y)[1] & 255) << 16) | \ + ((unsigned long)((y)[2] & 255) << 8) | \ + ((unsigned long)((y)[3] & 255)); } + + #define STORE64H(x, y) \ + { (y)[0] = (unsigned char)(((x) >> 56) & 255); (y)[1] = (unsigned char)(((x) >> 48) & 255); \ + (y)[2] = (unsigned char)(((x) >> 40) & 255); (y)[3] = (unsigned char)(((x) >> 32) & 255); \ + (y)[4] = (unsigned char)(((x) >> 24) & 255); (y)[5] = (unsigned char)(((x) >> 16) & 255); \ + (y)[6] = (unsigned char)(((x) >> 8) & 255); (y)[7] = (unsigned char)((x) & 255); } + + #define LOAD64H(x, y) \ + { x = (((ulong64)((y)[0] & 255)) << 56) | (((ulong64)((y)[1] & 255)) << 48) | \ + (((ulong64)((y)[2] & 255)) << 40) | (((ulong64)((y)[3] & 255)) << 32) | \ + (((ulong64)((y)[4] & 255)) << 24) | (((ulong64)((y)[5] & 255)) << 16) | \ + (((ulong64)((y)[6] & 255)) << 8) | (((ulong64)((y)[7] & 255))); } +#endif /* ENDIAN_NEUTRAL */ + +#ifdef ENDIAN_LITTLE + + #if !defined(LTC_NO_BSWAP) && (defined(INTEL_CC) || (defined(__GNUC__) && (defined(__DJGPP__) || defined(__CYGWIN__) || defined(__MINGW32__) || defined(__i386__) || defined(__x86_64__)))) + + #define STORE32H(x, y) \ + asm __volatile__ ( \ + "bswapl %0 \n\t" \ + "movl %0,(%1)\n\t" \ + "bswapl %0 \n\t" \ + ::"r" (x), "r" (y)); + + #define LOAD32H(x, y) \ + asm __volatile__ ( \ + "movl (%1),%0\n\t" \ + "bswapl %0\n\t" \ + : "=r" (x) : "r" (y)); + + #else + + #define STORE32H(x, y) \ + { (y)[0] = (unsigned char)(((x) >> 24) & 255); (y)[1] = (unsigned char)(((x) >> 16) & 255); \ + (y)[2] = (unsigned char)(((x) >> 8) & 255); (y)[3] = (unsigned char)((x) & 255); } + + #define LOAD32H(x, y) \ + { x = ((unsigned long)((y)[0] & 255) << 24) | \ + ((unsigned long)((y)[1] & 255) << 16) | \ + ((unsigned long)((y)[2] & 255) << 8) | \ + ((unsigned long)((y)[3] & 255)); } + #endif + + +/* x86_64 processor */ + #if !defined(LTC_NO_BSWAP) && (defined(__GNUC__) && defined(__x86_64__)) + + #define STORE64H(x, y) \ + asm __volatile__ ( \ + "bswapq %0 \n\t" \ + "movq %0,(%1)\n\t" \ + "bswapq %0 \n\t" \ + ::"r" (x), "r" (y)); + + #define LOAD64H(x, y) \ + asm __volatile__ ( \ + "movq (%1),%0\n\t" \ + "bswapq %0\n\t" \ + : "=r" (x) : "r" (y)); + + #else + + #define STORE64H(x, y) \ + { (y)[0] = (unsigned char)(((x) >> 56) & 255); (y)[1] = (unsigned char)(((x) >> 48) & 255); \ + (y)[2] = (unsigned char)(((x) >> 40) & 255); (y)[3] = (unsigned char)(((x) >> 32) & 255); \ + (y)[4] = (unsigned char)(((x) >> 24) & 255); (y)[5] = (unsigned char)(((x) >> 16) & 255); \ + (y)[6] = (unsigned char)(((x) >> 8) & 255); (y)[7] = (unsigned char)((x) & 255); } + + #define LOAD64H(x, y) \ + { x = (((ulong64)((y)[0] & 255)) << 56) | (((ulong64)((y)[1] & 255)) << 48) | \ + (((ulong64)((y)[2] & 255)) << 40) | (((ulong64)((y)[3] & 255)) << 32) | \ + (((ulong64)((y)[4] & 255)) << 24) | (((ulong64)((y)[5] & 255)) << 16) | \ + (((ulong64)((y)[6] & 255)) << 8) | (((ulong64)((y)[7] & 255))); } + #endif + + #ifdef ENDIAN_32BITWORD + + #define STORE32L(x, y) \ + { ulong32 __t = (x); XMEMCPY(y, &__t, 4); } + + #define LOAD32L(x, y) \ + XMEMCPY(&(x), y, 4); + + #define STORE64L(x, y) \ + { (y)[7] = (unsigned char)(((x) >> 56) & 255); (y)[6] = (unsigned char)(((x) >> 48) & 255); \ + (y)[5] = (unsigned char)(((x) >> 40) & 255); (y)[4] = (unsigned char)(((x) >> 32) & 255); \ + (y)[3] = (unsigned char)(((x) >> 24) & 255); (y)[2] = (unsigned char)(((x) >> 16) & 255); \ + (y)[1] = (unsigned char)(((x) >> 8) & 255); (y)[0] = (unsigned char)((x) & 255); } + + #define LOAD64L(x, y) \ + { x = (((ulong64)((y)[7] & 255)) << 56) | (((ulong64)((y)[6] & 255)) << 48) | \ + (((ulong64)((y)[5] & 255)) << 40) | (((ulong64)((y)[4] & 255)) << 32) | \ + (((ulong64)((y)[3] & 255)) << 24) | (((ulong64)((y)[2] & 255)) << 16) | \ + (((ulong64)((y)[1] & 255)) << 8) | (((ulong64)((y)[0] & 255))); } + + #else /* 64-bit words then */ + + #define STORE32L(x, y) \ + { ulong32 __t = (x); XMEMCPY(y, &__t, 4); } + + #define LOAD32L(x, y) \ + { XMEMCPY(&(x), y, 4); x &= 0xFFFFFFFF; } + + #define STORE64L(x, y) \ + { ulong64 __t = (x); XMEMCPY(y, &__t, 8); } + + #define LOAD64L(x, y) \ + { XMEMCPY(&(x), y, 8); } + #endif /* ENDIAN_64BITWORD */ +#endif /* ENDIAN_LITTLE */ + +#ifdef ENDIAN_BIG + #define STORE32L(x, y) \ + { (y)[3] = (unsigned char)(((x) >> 24) & 255); (y)[2] = (unsigned char)(((x) >> 16) & 255); \ + (y)[1] = (unsigned char)(((x) >> 8) & 255); (y)[0] = (unsigned char)((x) & 255); } + + #define LOAD32L(x, y) \ + { x = ((unsigned long)((y)[3] & 255) << 24) | \ + ((unsigned long)((y)[2] & 255) << 16) | \ + ((unsigned long)((y)[1] & 255) << 8) | \ + ((unsigned long)((y)[0] & 255)); } + + #define STORE64L(x, y) \ + { (y)[7] = (unsigned char)(((x) >> 56) & 255); (y)[6] = (unsigned char)(((x) >> 48) & 255); \ + (y)[5] = (unsigned char)(((x) >> 40) & 255); (y)[4] = (unsigned char)(((x) >> 32) & 255); \ + (y)[3] = (unsigned char)(((x) >> 24) & 255); (y)[2] = (unsigned char)(((x) >> 16) & 255); \ + (y)[1] = (unsigned char)(((x) >> 8) & 255); (y)[0] = (unsigned char)((x) & 255); } + + #define LOAD64L(x, y) \ + { x = (((ulong64)((y)[7] & 255)) << 56) | (((ulong64)((y)[6] & 255)) << 48) | \ + (((ulong64)((y)[5] & 255)) << 40) | (((ulong64)((y)[4] & 255)) << 32) | \ + (((ulong64)((y)[3] & 255)) << 24) | (((ulong64)((y)[2] & 255)) << 16) | \ + (((ulong64)((y)[1] & 255)) << 8) | (((ulong64)((y)[0] & 255))); } + + #ifdef ENDIAN_32BITWORD + + #define STORE32H(x, y) \ + { ulong32 __t = (x); XMEMCPY(y, &__t, 4); } + + #define LOAD32H(x, y) \ + XMEMCPY(&(x), y, 4); + + #define STORE64H(x, y) \ + { (y)[0] = (unsigned char)(((x) >> 56) & 255); (y)[1] = (unsigned char)(((x) >> 48) & 255); \ + (y)[2] = (unsigned char)(((x) >> 40) & 255); (y)[3] = (unsigned char)(((x) >> 32) & 255); \ + (y)[4] = (unsigned char)(((x) >> 24) & 255); (y)[5] = (unsigned char)(((x) >> 16) & 255); \ + (y)[6] = (unsigned char)(((x) >> 8) & 255); (y)[7] = (unsigned char)((x) & 255); } + + #define LOAD64H(x, y) \ + { x = (((ulong64)((y)[0] & 255)) << 56) | (((ulong64)((y)[1] & 255)) << 48) | \ + (((ulong64)((y)[2] & 255)) << 40) | (((ulong64)((y)[3] & 255)) << 32) | \ + (((ulong64)((y)[4] & 255)) << 24) | (((ulong64)((y)[5] & 255)) << 16) | \ + (((ulong64)((y)[6] & 255)) << 8) | (((ulong64)((y)[7] & 255))); } + + #else /* 64-bit words then */ + + #define STORE32H(x, y) \ + { ulong32 __t = (x); XMEMCPY(y, &__t, 4); } + + #define LOAD32H(x, y) \ + { XMEMCPY(&(x), y, 4); x &= 0xFFFFFFFF; } + + #define STORE64H(x, y) \ + { ulong64 __t = (x); XMEMCPY(y, &__t, 8); } + + #define LOAD64H(x, y) \ + { XMEMCPY(&(x), y, 8); } + #endif /* ENDIAN_64BITWORD */ +#endif /* ENDIAN_BIG */ + +#define BSWAP(x) \ + (((x >> 24) & 0x000000FFUL) | ((x << 24) & 0xFF000000UL) | \ + ((x >> 8) & 0x0000FF00UL) | ((x << 8) & 0x00FF0000UL)) + + +/* 32-bit Rotates */ +#if defined(_MSC_VER) + +/* instrinsic rotate */ + #include + #pragma intrinsic(_lrotr,_lrotl) + #define ROR(x, n) _lrotr(x, n) + #define ROL(x, n) _lrotl(x, n) + #define RORc(x, n) _lrotr(x, n) + #define ROLc(x, n) _lrotl(x, n) + +#elif !defined(__STRICT_ANSI__) && defined(__GNUC__) && (defined(__i386__) || defined(__x86_64__)) && !defined(INTEL_CC) && !defined(LTC_NO_ASM) + +static inline unsigned ROL(unsigned word, int i) { + asm ("roll %%cl,%0" + : "=r" (word) + : "0" (word), "c" (i)); + return word; +} + +static inline unsigned ROR(unsigned word, int i) { + asm ("rorl %%cl,%0" + : "=r" (word) + : "0" (word), "c" (i)); + return word; +} + + #ifndef LTC_NO_ROLC + +static inline unsigned ROLc(unsigned word, const int i) { + asm ("roll %2,%0" + : "=r" (word) + : "0" (word), "I" (i)); + return word; +} + +static inline unsigned RORc(unsigned word, const int i) { + asm ("rorl %2,%0" + : "=r" (word) + : "0" (word), "I" (i)); + return word; +} + + #else + + #define ROLc ROL + #define RORc ROR + #endif + +#elif !defined(__STRICT_ANSI__) && defined(LTC_PPC32) + +static inline unsigned ROL(unsigned word, int i) { + asm ("rotlw %0,%0,%2" + : "=r" (word) + : "0" (word), "r" (i)); + return word; +} + +static inline unsigned ROR(unsigned word, int i) { + asm ("rotlw %0,%0,%2" + : "=r" (word) + : "0" (word), "r" (32 - i)); + return word; +} + + #ifndef LTC_NO_ROLC + +static inline unsigned ROLc(unsigned word, const int i) { + asm ("rotlwi %0,%0,%2" + : "=r" (word) + : "0" (word), "I" (i)); + return word; +} + +static inline unsigned RORc(unsigned word, const int i) { + asm ("rotrwi %0,%0,%2" + : "=r" (word) + : "0" (word), "I" (i)); + return word; +} + + #else + + #define ROLc ROL + #define RORc ROR + #endif + + +#else + +/* rotates the hard way */ + #define ROL(x, y) ((((unsigned long)(x) << (unsigned long)((y) & 31)) | (((unsigned long)(x) & 0xFFFFFFFFUL) >> (unsigned long)(32 - ((y) & 31)))) & 0xFFFFFFFFUL) + #define ROR(x, y) (((((unsigned long)(x) & 0xFFFFFFFFUL) >> (unsigned long)((y) & 31)) | ((unsigned long)(x) << (unsigned long)(32 - ((y) & 31)))) & 0xFFFFFFFFUL) + #define ROLc(x, y) ((((unsigned long)(x) << (unsigned long)((y) & 31)) | (((unsigned long)(x) & 0xFFFFFFFFUL) >> (unsigned long)(32 - ((y) & 31)))) & 0xFFFFFFFFUL) + #define RORc(x, y) (((((unsigned long)(x) & 0xFFFFFFFFUL) >> (unsigned long)((y) & 31)) | ((unsigned long)(x) << (unsigned long)(32 - ((y) & 31)))) & 0xFFFFFFFFUL) +#endif + + +/* 64-bit Rotates */ +#if !defined(__STRICT_ANSI__) && defined(__GNUC__) && defined(__x86_64__) && !defined(LTC_NO_ASM) + +static inline unsigned long ROL64(unsigned long word, int i) { + asm ("rolq %%cl,%0" + : "=r" (word) + : "0" (word), "c" (i)); + return word; +} + +static inline unsigned long ROR64(unsigned long word, int i) { + asm ("rorq %%cl,%0" + : "=r" (word) + : "0" (word), "c" (i)); + return word; +} + + #ifndef LTC_NO_ROLC + +static inline unsigned long ROL64c(unsigned long word, const int i) { + asm ("rolq %2,%0" + : "=r" (word) + : "0" (word), "J" (i)); + return word; +} + +static inline unsigned long ROR64c(unsigned long word, const int i) { + asm ("rorq %2,%0" + : "=r" (word) + : "0" (word), "J" (i)); + return word; +} + + #else /* LTC_NO_ROLC */ + + #define ROL64c ROL64 + #define ROR64c ROR64 + #endif + +#else /* Not x86_64 */ + + #define ROL64(x, y) \ + ((((x) << ((ulong64)(y) & 63)) | \ + (((x) & CONST64(0xFFFFFFFFFFFFFFFF)) >> ((ulong64)64 - ((y) & 63)))) & CONST64(0xFFFFFFFFFFFFFFFF)) + + #define ROR64(x, y) \ + (((((x) & CONST64(0xFFFFFFFFFFFFFFFF)) >> ((ulong64)(y) & CONST64(63))) | \ + ((x) << ((ulong64)(64 - ((y) & CONST64(63)))))) & CONST64(0xFFFFFFFFFFFFFFFF)) + + #define ROL64c(x, y) \ + ((((x) << ((ulong64)(y) & 63)) | \ + (((x) & CONST64(0xFFFFFFFFFFFFFFFF)) >> ((ulong64)64 - ((y) & 63)))) & CONST64(0xFFFFFFFFFFFFFFFF)) + + #define ROR64c(x, y) \ + (((((x) & CONST64(0xFFFFFFFFFFFFFFFF)) >> ((ulong64)(y) & CONST64(63))) | \ + ((x) << ((ulong64)(64 - ((y) & CONST64(63)))))) & CONST64(0xFFFFFFFFFFFFFFFF)) +#endif + +#ifndef MAX + #define MAX(x, y) (((x) > (y)) ? (x) : (y)) +#endif + +#ifndef MIN + #define MIN(x, y) (((x) < (y)) ? (x) : (y)) +#endif + +/* extract a byte portably */ +#ifdef _MSC_VER + #define byte(x, n) ((unsigned char)((x) >> (8 * (n)))) +#else + #define byte(x, n) (((x) >> (8 * (n))) & 255) +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_macros.h,v $ */ +/* $Revision: 1.15 $ */ +/* $Date: 2006/11/29 23:43:57 $ */ + +/* ---- SYMMETRIC KEY STUFF ----- + * + * We put each of the ciphers scheduled keys in their own structs then we put all of + * the key formats in one union. This makes the function prototypes easier to use. + */ +#ifdef LTC_BLOWFISH +struct blowfish_key { + ulong32 S[4][256]; + ulong32 K[18]; +}; +#endif + +#ifdef LTC_RC5 +struct rc5_key { + int rounds; + ulong32 K[50]; +}; +#endif + +#ifdef LTC_RC6 +struct rc6_key { + ulong32 K[44]; +}; +#endif + +#ifdef LTC_SAFERP +struct saferp_key { + unsigned char K[33][16]; + long rounds; +}; +#endif + +#ifdef LTC_RIJNDAEL +struct rijndael_key { + ulong32 eK[60], dK[60]; + int Nr; +}; +#endif + +#ifdef LTC_KSEED +struct kseed_key { + ulong32 K[32], dK[32]; +}; +#endif + +#ifdef LTC_KASUMI +struct kasumi_key { + ulong32 KLi1[8], KLi2[8], + KOi1[8], KOi2[8], KOi3[8], + KIi1[8], KIi2[8], KIi3[8]; +}; +#endif + +#ifdef LTC_XTEA +struct xtea_key { + unsigned long A[32], B[32]; +}; +#endif + +#ifdef LTC_TWOFISH + #ifndef LTC_TWOFISH_SMALL +struct twofish_key { + ulong32 S[4][256], K[40]; +}; + #else +struct twofish_key { + ulong32 K[40]; + unsigned char S[32], start; +}; + #endif +#endif + +#ifdef LTC_SAFER + #define LTC_SAFER_K64_DEFAULT_NOF_ROUNDS 6 + #define LTC_SAFER_K128_DEFAULT_NOF_ROUNDS 10 + #define LTC_SAFER_SK64_DEFAULT_NOF_ROUNDS 8 + #define LTC_SAFER_SK128_DEFAULT_NOF_ROUNDS 10 + #define LTC_SAFER_MAX_NOF_ROUNDS 13 + #define LTC_SAFER_BLOCK_LEN 8 + #define LTC_SAFER_KEY_LEN (1 + LTC_SAFER_BLOCK_LEN * (1 + 2 * LTC_SAFER_MAX_NOF_ROUNDS)) +typedef unsigned char safer_block_t[LTC_SAFER_BLOCK_LEN]; +typedef unsigned char safer_key_t[LTC_SAFER_KEY_LEN]; +struct safer_key { + safer_key_t key; +}; +#endif + +#ifdef LTC_RC2 +struct rc2_key { + unsigned xkey[64]; +}; +#endif + +#ifdef LTC_DES +struct des_key { + ulong32 ek[32], dk[32]; +}; + +struct des3_key { + ulong32 ek[3][32], dk[3][32]; +}; +#endif + +#ifdef LTC_CAST5 +struct cast5_key { + ulong32 K[32], keylen; +}; +#endif + +#ifdef LTC_NOEKEON +struct noekeon_key { + ulong32 K[4], dK[4]; +}; +#endif + +#ifdef LTC_SKIPJACK +struct skipjack_key { + unsigned char key[10]; +}; +#endif + +#ifdef LTC_KHAZAD +struct khazad_key { + ulong64 roundKeyEnc[8 + 1]; + ulong64 roundKeyDec[8 + 1]; +}; +#endif + +#ifdef LTC_ANUBIS +struct anubis_key { + int keyBits; + int R; + ulong32 roundKeyEnc[18 + 1][4]; + ulong32 roundKeyDec[18 + 1][4]; +}; +#endif + +#ifdef LTC_MULTI2 +struct multi2_key { + int N; + ulong32 uk[8]; +}; +#endif + +typedef union Symmetric_key { +#ifdef LTC_DES + struct des_key des; + struct des3_key des3; +#endif +#ifdef LTC_RC2 + struct rc2_key rc2; +#endif +#ifdef LTC_SAFER + struct safer_key safer; +#endif +#ifdef LTC_TWOFISH + struct twofish_key twofish; +#endif +#ifdef LTC_BLOWFISH + struct blowfish_key blowfish; +#endif +#ifdef LTC_RC5 + struct rc5_key rc5; +#endif +#ifdef LTC_RC6 + struct rc6_key rc6; +#endif +#ifdef LTC_SAFERP + struct saferp_key saferp; +#endif +#ifdef LTC_RIJNDAEL + struct rijndael_key rijndael; +#endif +#ifdef LTC_XTEA + struct xtea_key xtea; +#endif +#ifdef LTC_CAST5 + struct cast5_key cast5; +#endif +#ifdef LTC_NOEKEON + struct noekeon_key noekeon; +#endif +#ifdef LTC_SKIPJACK + struct skipjack_key skipjack; +#endif +#ifdef LTC_KHAZAD + struct khazad_key khazad; +#endif +#ifdef LTC_ANUBIS + struct anubis_key anubis; +#endif +#ifdef LTC_KSEED + struct kseed_key kseed; +#endif +#ifdef LTC_KASUMI + struct kasumi_key kasumi; +#endif +#ifdef LTC_MULTI2 + struct multi2_key multi2; +#endif + void *data; +} symmetric_key; + +#ifdef LTC_ECB_MODE +/** A block cipher ECB structure */ +typedef struct { + /** The index of the cipher chosen */ + int cipher, + /** The block size of the given cipher */ + blocklen; + /** The scheduled key */ + symmetric_key key; +} symmetric_ECB; +#endif + +#ifdef LTC_CFB_MODE +/** A block cipher CFB structure */ +typedef struct { + /** The index of the cipher chosen */ + int cipher, + /** The block size of the given cipher */ + blocklen, + /** The padding offset */ + padlen; + /** The current IV */ + unsigned char IV[MAXBLOCKSIZE], + /** The pad used to encrypt/decrypt */ + pad[MAXBLOCKSIZE]; + /** The scheduled key */ + symmetric_key key; +} symmetric_CFB; +#endif + +#ifdef LTC_OFB_MODE +/** A block cipher OFB structure */ +typedef struct { + /** The index of the cipher chosen */ + int cipher, + /** The block size of the given cipher */ + blocklen, + /** The padding offset */ + padlen; + /** The current IV */ + unsigned char IV[MAXBLOCKSIZE]; + /** The scheduled key */ + symmetric_key key; +} symmetric_OFB; +#endif + +#ifdef LTC_CBC_MODE +/** A block cipher CBC structure */ +typedef struct { + /** The index of the cipher chosen */ + int cipher, + /** The block size of the given cipher */ + blocklen; + /** The current IV */ + unsigned char IV[MAXBLOCKSIZE]; + /** The scheduled key */ + symmetric_key key; +} symmetric_CBC; +#endif + + +#ifdef LTC_CTR_MODE +/** A block cipher CTR structure */ +typedef struct { + /** The index of the cipher chosen */ + int cipher, + /** The block size of the given cipher */ + blocklen, + /** The padding offset */ + padlen, + /** The mode (endianess) of the CTR, 0==little, 1==big */ + mode, + /** counter width */ + ctrlen; + + /** The counter */ + unsigned char ctr[MAXBLOCKSIZE], + /** The pad used to encrypt/decrypt */ + pad[MAXBLOCKSIZE]; + /** The scheduled key */ + symmetric_key key; +} symmetric_CTR; +#endif + + +#ifdef LTC_LRW_MODE +/** A LRW structure */ +typedef struct { + /** The index of the cipher chosen (must be a 128-bit block cipher) */ + int cipher; + + /** The current IV */ + unsigned char IV[16], + + /** the tweak key */ + tweak[16], + + /** The current pad, it's the product of the first 15 bytes against the tweak key */ + pad[16]; + + /** The scheduled symmetric key */ + symmetric_key key; + + #ifdef LRW_TABLES + /** The pre-computed multiplication table */ + unsigned char PC[16][256][16]; + #endif +} symmetric_LRW; +#endif + +#ifdef LTC_F8_MODE +/** A block cipher F8 structure */ +typedef struct { + /** The index of the cipher chosen */ + int cipher, + /** The block size of the given cipher */ + blocklen, + /** The padding offset */ + padlen; + /** The current IV */ + unsigned char IV[MAXBLOCKSIZE], + MIV[MAXBLOCKSIZE]; + /** Current block count */ + ulong32 blockcnt; + /** The scheduled key */ + symmetric_key key; +} symmetric_F8; +#endif + + +/** cipher descriptor table, last entry has "name == NULL" to mark the end of table */ +extern struct ltc_cipher_descriptor { + /** name of cipher */ + char *name; + /** internal ID */ + unsigned char ID; + /** min keysize (octets) */ + int min_key_length, + /** max keysize (octets) */ + max_key_length, + /** block size (octets) */ + block_length, + /** default number of rounds */ + default_rounds; + + /** Setup the cipher + @param key The input symmetric key + @param keylen The length of the input key (octets) + @param num_rounds The requested number of rounds (0==default) + @param skey [out] The destination of the scheduled key + @return CRYPT_OK if successful + */ + int (*setup)(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); + + /** Encrypt a block + @param pt The plaintext + @param ct [out] The ciphertext + @param skey The scheduled key + @return CRYPT_OK if successful + */ + int (*ecb_encrypt)(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); + + /** Decrypt a block + @param ct The ciphertext + @param pt [out] The plaintext + @param skey The scheduled key + @return CRYPT_OK if successful + */ + int (*ecb_decrypt)(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); + + /** Test the block cipher + @return CRYPT_OK if successful, CRYPT_NOP if self-testing has been disabled + */ + int (*test)(void); + + /** Terminate the context + @param skey The scheduled key + */ + void (*done)(symmetric_key *skey); + + /** Determine a key size + @param keysize [in/out] The size of the key desired and the suggested size + @return CRYPT_OK if successful + */ + int (*keysize)(int *keysize); + +/** Accelerators **/ + + /** Accelerated ECB encryption + @param pt Plaintext + @param ct Ciphertext + @param blocks The number of complete blocks to process + @param skey The scheduled key context + @return CRYPT_OK if successful + */ + int (*accel_ecb_encrypt)(const unsigned char *pt, unsigned char *ct, unsigned long blocks, symmetric_key *skey); + + /** Accelerated ECB decryption + @param pt Plaintext + @param ct Ciphertext + @param blocks The number of complete blocks to process + @param skey The scheduled key context + @return CRYPT_OK if successful + */ + int (*accel_ecb_decrypt)(const unsigned char *ct, unsigned char *pt, unsigned long blocks, symmetric_key *skey); + + /** Accelerated CBC encryption + @param pt Plaintext + @param ct Ciphertext + @param blocks The number of complete blocks to process + @param IV The initial value (input/output) + @param skey The scheduled key context + @return CRYPT_OK if successful + */ + int (*accel_cbc_encrypt)(const unsigned char *pt, unsigned char *ct, unsigned long blocks, unsigned char *IV, symmetric_key *skey); + + /** Accelerated CBC decryption + @param pt Plaintext + @param ct Ciphertext + @param blocks The number of complete blocks to process + @param IV The initial value (input/output) + @param skey The scheduled key context + @return CRYPT_OK if successful + */ + int (*accel_cbc_decrypt)(const unsigned char *ct, unsigned char *pt, unsigned long blocks, unsigned char *IV, symmetric_key *skey); + + /** Accelerated CTR encryption + @param pt Plaintext + @param ct Ciphertext + @param blocks The number of complete blocks to process + @param IV The initial value (input/output) + @param mode little or big endian counter (mode=0 or mode=1) + @param skey The scheduled key context + @return CRYPT_OK if successful + */ + int (*accel_ctr_encrypt)(const unsigned char *pt, unsigned char *ct, unsigned long blocks, unsigned char *IV, int mode, symmetric_key *skey); + + /** Accelerated LRW + @param pt Plaintext + @param ct Ciphertext + @param blocks The number of complete blocks to process + @param IV The initial value (input/output) + @param tweak The LRW tweak + @param skey The scheduled key context + @return CRYPT_OK if successful + */ + int (*accel_lrw_encrypt)(const unsigned char *pt, unsigned char *ct, unsigned long blocks, unsigned char *IV, const unsigned char *tweak, symmetric_key *skey); + + /** Accelerated LRW + @param ct Ciphertext + @param pt Plaintext + @param blocks The number of complete blocks to process + @param IV The initial value (input/output) + @param tweak The LRW tweak + @param skey The scheduled key context + @return CRYPT_OK if successful + */ + int (*accel_lrw_decrypt)(const unsigned char *ct, unsigned char *pt, unsigned long blocks, unsigned char *IV, const unsigned char *tweak, symmetric_key *skey); + + /** Accelerated CCM packet (one-shot) + @param key The secret key to use + @param keylen The length of the secret key (octets) + @param uskey A previously scheduled key [optional can be NULL] + @param nonce The session nonce [use once] + @param noncelen The length of the nonce + @param header The header for the session + @param headerlen The length of the header (octets) + @param pt [out] The plaintext + @param ptlen The length of the plaintext (octets) + @param ct [out] The ciphertext + @param tag [out] The destination tag + @param taglen [in/out] The max size and resulting size of the authentication tag + @param direction Encrypt or Decrypt direction (0 or 1) + @return CRYPT_OK if successful + */ + int (*accel_ccm_memory)( + const unsigned char *key, unsigned long keylen, + symmetric_key *uskey, + const unsigned char *nonce, unsigned long noncelen, + const unsigned char *header, unsigned long headerlen, + unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + unsigned char *tag, unsigned long *taglen, + int direction); + + /** Accelerated GCM packet (one shot) + @param key The secret key + @param keylen The length of the secret key + @param IV The initial vector + @param IVlen The length of the initial vector + @param adata The additional authentication data (header) + @param adatalen The length of the adata + @param pt The plaintext + @param ptlen The length of the plaintext (ciphertext length is the same) + @param ct The ciphertext + @param tag [out] The MAC tag + @param taglen [in/out] The MAC tag length + @param direction Encrypt or Decrypt mode (GCM_ENCRYPT or GCM_DECRYPT) + @return CRYPT_OK on success + */ + int (*accel_gcm_memory)( + const unsigned char *key, unsigned long keylen, + const unsigned char *IV, unsigned long IVlen, + const unsigned char *adata, unsigned long adatalen, + unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + unsigned char *tag, unsigned long *taglen, + int direction); + + /** Accelerated one shot LTC_OMAC + @param key The secret key + @param keylen The key length (octets) + @param in The message + @param inlen Length of message (octets) + @param out [out] Destination for tag + @param outlen [in/out] Initial and final size of out + @return CRYPT_OK on success + */ + int (*omac_memory)( + const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); + + /** Accelerated one shot XCBC + @param key The secret key + @param keylen The key length (octets) + @param in The message + @param inlen Length of message (octets) + @param out [out] Destination for tag + @param outlen [in/out] Initial and final size of out + @return CRYPT_OK on success + */ + int (*xcbc_memory)( + const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); + + /** Accelerated one shot F9 + @param key The secret key + @param keylen The key length (octets) + @param in The message + @param inlen Length of message (octets) + @param out [out] Destination for tag + @param outlen [in/out] Initial and final size of out + @return CRYPT_OK on success + @remark Requires manual padding + */ + int (*f9_memory)( + const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +} cipher_descriptor[]; + +#ifdef LTC_BLOWFISH +int blowfish_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int blowfish_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int blowfish_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int blowfish_test(void); +void blowfish_done(symmetric_key *skey); +int blowfish_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor blowfish_desc; +#endif + +#ifdef LTC_RC5 +int rc5_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int rc5_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int rc5_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int rc5_test(void); +void rc5_done(symmetric_key *skey); +int rc5_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor rc5_desc; +#endif + +#ifdef LTC_RC6 +int rc6_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int rc6_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int rc6_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int rc6_test(void); +void rc6_done(symmetric_key *skey); +int rc6_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor rc6_desc; +#endif + +#ifdef LTC_RC2 +int rc2_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int rc2_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int rc2_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int rc2_test(void); +void rc2_done(symmetric_key *skey); +int rc2_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor rc2_desc; +#endif + +#ifdef LTC_SAFERP +int saferp_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int saferp_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int saferp_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int saferp_test(void); +void saferp_done(symmetric_key *skey); +int saferp_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor saferp_desc; +#endif + +#ifdef LTC_SAFER +int safer_k64_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int safer_sk64_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int safer_k128_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int safer_sk128_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int safer_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *key); +int safer_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *key); +int safer_k64_test(void); +int safer_sk64_test(void); +int safer_sk128_test(void); +void safer_done(symmetric_key *skey); +int safer_64_keysize(int *keysize); +int safer_128_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor safer_k64_desc, safer_k128_desc, safer_sk64_desc, safer_sk128_desc; +#endif + +#ifdef LTC_RIJNDAEL + +/* make aes an alias */ + #define aes_setup rijndael_setup + #define aes_ecb_encrypt rijndael_ecb_encrypt + #define aes_ecb_decrypt rijndael_ecb_decrypt + #define aes_test rijndael_test + #define aes_done rijndael_done + #define aes_keysize rijndael_keysize + + #define aes_enc_setup rijndael_enc_setup + #define aes_enc_ecb_encrypt rijndael_enc_ecb_encrypt + #define aes_enc_keysize rijndael_enc_keysize + +int rijndael_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int rijndael_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int rijndael_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int rijndael_test(void); +void rijndael_done(symmetric_key *skey); +int rijndael_keysize(int *keysize); +int rijndael_enc_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int rijndael_enc_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +void rijndael_enc_done(symmetric_key *skey); +int rijndael_enc_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor rijndael_desc, aes_desc; +extern const struct ltc_cipher_descriptor rijndael_enc_desc, aes_enc_desc; +#endif + +#ifdef LTC_XTEA +int xtea_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int xtea_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int xtea_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int xtea_test(void); +void xtea_done(symmetric_key *skey); +int xtea_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor xtea_desc; +#endif + +#ifdef LTC_TWOFISH +int twofish_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int twofish_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int twofish_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int twofish_test(void); +void twofish_done(symmetric_key *skey); +int twofish_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor twofish_desc; +#endif + +#ifdef LTC_DES +int des_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int des_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int des_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int des_test(void); +void des_done(symmetric_key *skey); +int des_keysize(int *keysize); +int des3_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int des3_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int des3_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int des3_test(void); +void des3_done(symmetric_key *skey); +int des3_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor des_desc, des3_desc; +#endif + +#ifdef LTC_CAST5 +int cast5_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int cast5_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int cast5_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int cast5_test(void); +void cast5_done(symmetric_key *skey); +int cast5_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor cast5_desc; +#endif + +#ifdef LTC_NOEKEON +int noekeon_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int noekeon_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int noekeon_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int noekeon_test(void); +void noekeon_done(symmetric_key *skey); +int noekeon_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor noekeon_desc; +#endif + +#ifdef LTC_SKIPJACK +int skipjack_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int skipjack_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int skipjack_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int skipjack_test(void); +void skipjack_done(symmetric_key *skey); +int skipjack_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor skipjack_desc; +#endif + +#ifdef LTC_KHAZAD +int khazad_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int khazad_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int khazad_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int khazad_test(void); +void khazad_done(symmetric_key *skey); +int khazad_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor khazad_desc; +#endif + +#ifdef LTC_ANUBIS +int anubis_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int anubis_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int anubis_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int anubis_test(void); +void anubis_done(symmetric_key *skey); +int anubis_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor anubis_desc; +#endif + +#ifdef LTC_KSEED +int kseed_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int kseed_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int kseed_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int kseed_test(void); +void kseed_done(symmetric_key *skey); +int kseed_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor kseed_desc; +#endif + +#ifdef LTC_KASUMI +int kasumi_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int kasumi_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int kasumi_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int kasumi_test(void); +void kasumi_done(symmetric_key *skey); +int kasumi_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor kasumi_desc; +#endif + + +#ifdef LTC_MULTI2 +int multi2_setup(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey); +int multi2_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey); +int multi2_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey); +int multi2_test(void); +void multi2_done(symmetric_key *skey); +int multi2_keysize(int *keysize); + +extern const struct ltc_cipher_descriptor multi2_desc; +#endif + +#ifdef LTC_ECB_MODE +int ecb_start(int cipher, const unsigned char *key, + int keylen, int num_rounds, symmetric_ECB *ecb); +int ecb_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_ECB *ecb); +int ecb_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_ECB *ecb); +int ecb_done(symmetric_ECB *ecb); +#endif + +#ifdef LTC_CFB_MODE +int cfb_start(int cipher, const unsigned char *IV, const unsigned char *key, + int keylen, int num_rounds, symmetric_CFB *cfb); +int cfb_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_CFB *cfb); +int cfb_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_CFB *cfb); +int cfb_getiv(unsigned char *IV, unsigned long *len, symmetric_CFB *cfb); +int cfb_setiv(const unsigned char *IV, unsigned long len, symmetric_CFB *cfb); +int cfb_done(symmetric_CFB *cfb); +#endif + +#ifdef LTC_OFB_MODE +int ofb_start(int cipher, const unsigned char *IV, const unsigned char *key, + int keylen, int num_rounds, symmetric_OFB *ofb); +int ofb_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_OFB *ofb); +int ofb_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_OFB *ofb); +int ofb_getiv(unsigned char *IV, unsigned long *len, symmetric_OFB *ofb); +int ofb_setiv(const unsigned char *IV, unsigned long len, symmetric_OFB *ofb); +int ofb_done(symmetric_OFB *ofb); +#endif + +#ifdef LTC_CBC_MODE +int cbc_start(int cipher, const unsigned char *IV, const unsigned char *key, + int keylen, int num_rounds, symmetric_CBC *cbc); +int cbc_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_CBC *cbc); +int cbc_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_CBC *cbc); +int cbc_getiv(unsigned char *IV, unsigned long *len, symmetric_CBC *cbc); +int cbc_setiv(const unsigned char *IV, unsigned long len, symmetric_CBC *cbc); +int cbc_done(symmetric_CBC *cbc); +#endif + +#ifdef LTC_CTR_MODE + + #define CTR_COUNTER_LITTLE_ENDIAN 0x0000 + #define CTR_COUNTER_BIG_ENDIAN 0x1000 + #define LTC_CTR_RFC3686 0x2000 + +int ctr_start(int cipher, + const unsigned char *IV, + const unsigned char *key, int keylen, + int num_rounds, int ctr_mode, + symmetric_CTR *ctr); +int ctr_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_CTR *ctr); +int ctr_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_CTR *ctr); +int ctr_getiv(unsigned char *IV, unsigned long *len, symmetric_CTR *ctr); +int ctr_setiv(const unsigned char *IV, unsigned long len, symmetric_CTR *ctr); +int ctr_done(symmetric_CTR *ctr); +int ctr_test(void); +#endif + +#ifdef LTC_LRW_MODE + + #define LRW_ENCRYPT 0 + #define LRW_DECRYPT 1 + +int lrw_start(int cipher, + const unsigned char *IV, + const unsigned char *key, int keylen, + const unsigned char *tweak, + int num_rounds, + symmetric_LRW *lrw); +int lrw_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_LRW *lrw); +int lrw_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_LRW *lrw); +int lrw_getiv(unsigned char *IV, unsigned long *len, symmetric_LRW *lrw); +int lrw_setiv(const unsigned char *IV, unsigned long len, symmetric_LRW *lrw); +int lrw_done(symmetric_LRW *lrw); +int lrw_test(void); + +/* don't call */ +int lrw_process(const unsigned char *pt, unsigned char *ct, unsigned long len, int mode, symmetric_LRW *lrw); +#endif + +#ifdef LTC_F8_MODE +int f8_start(int cipher, const unsigned char *IV, + const unsigned char *key, int keylen, + const unsigned char *salt_key, int skeylen, + int num_rounds, symmetric_F8 *f8); +int f8_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_F8 *f8); +int f8_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_F8 *f8); +int f8_getiv(unsigned char *IV, unsigned long *len, symmetric_F8 *f8); +int f8_setiv(const unsigned char *IV, unsigned long len, symmetric_F8 *f8); +int f8_done(symmetric_F8 *f8); +int f8_test_mode(void); +#endif + +#ifdef LTC_XTS_MODE +typedef struct { + symmetric_key key1, key2; + int cipher; +} symmetric_xts; + +int xts_start(int cipher, + const unsigned char *key1, + const unsigned char *key2, + unsigned long keylen, + int num_rounds, + symmetric_xts *xts); + +int xts_encrypt( + const unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + const unsigned char *tweak, + symmetric_xts *xts); +int xts_decrypt( + const unsigned char *ct, unsigned long ptlen, + unsigned char *pt, + const unsigned char *tweak, + symmetric_xts *xts); + +void xts_done(symmetric_xts *xts); +int xts_test(void); +void xts_mult_x(unsigned char *I); +#endif + +int find_cipher(const char *name); +int find_cipher_any(const char *name, int blocklen, int keylen); +int find_cipher_id(unsigned char ID); +int register_cipher(const struct ltc_cipher_descriptor *cipher); +int unregister_cipher(const struct ltc_cipher_descriptor *cipher); +int cipher_is_valid(int idx); + +LTC_MUTEX_PROTO(ltc_cipher_mutex) + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_cipher.h,v $ */ +/* $Revision: 1.54 $ */ +/* $Date: 2007/05/12 14:37:41 $ */ + +#define LTC_SHA1 +/* ---- HASH FUNCTIONS ---- */ +#ifdef LTC_SHA512 +struct sha512_state { + ulong64 length, state[8]; + unsigned long curlen; + unsigned char buf[128]; +}; +#endif + +#ifdef LTC_SHA256 +struct sha256_state { + ulong64 length; + ulong32 state[8], curlen; + unsigned char buf[64]; +}; +#endif + +#ifdef LTC_SHA1 +struct sha1_state { + ulong64 length; + ulong32 state[5], curlen; + unsigned char buf[64]; +}; +#endif + +#ifdef LTC_MD5 +struct md5_state { + ulong64 length; + ulong32 state[4], curlen; + unsigned char buf[64]; +}; +#endif + +#ifdef LTC_MD4 +struct md4_state { + ulong64 length; + ulong32 state[4], curlen; + unsigned char buf[64]; +}; +#endif + +#ifdef LTC_TIGER +struct tiger_state { + ulong64 state[3], length; + unsigned long curlen; + unsigned char buf[64]; +}; +#endif + +#ifdef LTC_MD2 +struct md2_state { + unsigned char chksum[16], X[48], buf[16]; + unsigned long curlen; +}; +#endif + +#ifdef LTC_RIPEMD128 +struct rmd128_state { + ulong64 length; + unsigned char buf[64]; + ulong32 curlen, state[4]; +}; +#endif + +#ifdef LTC_RIPEMD160 +struct rmd160_state { + ulong64 length; + unsigned char buf[64]; + ulong32 curlen, state[5]; +}; +#endif + +#ifdef LTC_RIPEMD256 +struct rmd256_state { + ulong64 length; + unsigned char buf[64]; + ulong32 curlen, state[8]; +}; +#endif + +#ifdef LTC_RIPEMD320 +struct rmd320_state { + ulong64 length; + unsigned char buf[64]; + ulong32 curlen, state[10]; +}; +#endif + +#ifdef LTC_WHIRLPOOL +struct whirlpool_state { + ulong64 length, state[8]; + unsigned char buf[64]; + ulong32 curlen; +}; +#endif + +#ifdef LTC_CHC_HASH +struct chc_state { + ulong64 length; + unsigned char state[MAXBLOCKSIZE], buf[MAXBLOCKSIZE]; + ulong32 curlen; +}; +#endif + +typedef union Hash_state { + char dummy[1]; +#ifdef LTC_CHC_HASH + struct chc_state chc; +#endif +#ifdef LTC_WHIRLPOOL + struct whirlpool_state whirlpool; +#endif +#ifdef LTC_SHA512 + struct sha512_state sha512; +#endif +#ifdef LTC_SHA256 + struct sha256_state sha256; +#endif +#ifdef LTC_SHA1 + struct sha1_state sha1; +#endif +#ifdef LTC_MD5 + struct md5_state md5; +#endif +#ifdef LTC_MD4 + struct md4_state md4; +#endif +#ifdef LTC_MD2 + struct md2_state md2; +#endif +#ifdef LTC_TIGER + struct tiger_state tiger; +#endif +#ifdef LTC_RIPEMD128 + struct rmd128_state rmd128; +#endif +#ifdef LTC_RIPEMD160 + struct rmd160_state rmd160; +#endif +#ifdef LTC_RIPEMD256 + struct rmd256_state rmd256; +#endif +#ifdef LTC_RIPEMD320 + struct rmd320_state rmd320; +#endif + void *data; +} hash_state; + +/** hash descriptor */ +extern struct ltc_hash_descriptor { + /** name of hash */ + char *name; + /** internal ID */ + unsigned char ID; + /** Size of digest in octets */ + unsigned long hashsize; + /** Input block size in octets */ + unsigned long blocksize; + /** ASN.1 OID */ + unsigned long OID[16]; + /** Length of DER encoding */ + unsigned long OIDlen; + + /** Init a hash state + @param hash The hash to initialize + @return CRYPT_OK if successful + */ + int (*init)(hash_state *hash); + + /** Process a block of data + @param hash The hash state + @param in The data to hash + @param inlen The length of the data (octets) + @return CRYPT_OK if successful + */ + int (*process)(hash_state *hash, const unsigned char *in, unsigned long inlen); + + /** Produce the digest and store it + @param hash The hash state + @param out [out] The destination of the digest + @return CRYPT_OK if successful + */ + int (*done)(hash_state *hash, unsigned char *out); + + /** Self-test + @return CRYPT_OK if successful, CRYPT_NOP if self-tests have been disabled + */ + int (*test)(void); + + /* accelerated hmac callback: if you need to-do multiple packets just use the generic hmac_memory and provide a hash callback */ + int (*hmac_block)(const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +} hash_descriptor[]; + +#ifdef LTC_CHC_HASH +int chc_register(int cipher); +int chc_init(hash_state *md); +int chc_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int chc_done(hash_state *md, unsigned char *hash); +int chc_test(void); + +extern const struct ltc_hash_descriptor chc_desc; +#endif + +#ifdef LTC_WHIRLPOOL +int whirlpool_init(hash_state *md); +int whirlpool_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int whirlpool_done(hash_state *md, unsigned char *hash); +int whirlpool_test(void); + +extern const struct ltc_hash_descriptor whirlpool_desc; +#endif + +#ifdef LTC_SHA512 +int sha512_init(hash_state *md); +int sha512_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int sha512_done(hash_state *md, unsigned char *hash); +int sha512_test(void); + +extern const struct ltc_hash_descriptor sha512_desc; +#endif + +#ifdef LTC_SHA384 + #ifndef LTC_SHA512 + #error LTC_SHA512 is required for LTC_SHA384 + #endif +int sha384_init(hash_state *md); + + #define sha384_process sha512_process +int sha384_done(hash_state *md, unsigned char *hash); +int sha384_test(void); + +extern const struct ltc_hash_descriptor sha384_desc; +#endif + +#ifdef LTC_SHA256 +int sha256_init(hash_state *md); +int sha256_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int sha256_done(hash_state *md, unsigned char *hash); +int sha256_test(void); + +extern const struct ltc_hash_descriptor sha256_desc; + + #ifdef LTC_SHA224 + #ifndef LTC_SHA256 + #error LTC_SHA256 is required for LTC_SHA224 + #endif +int sha224_init(hash_state *md); + + #define sha224_process sha256_process +int sha224_done(hash_state *md, unsigned char *hash); +int sha224_test(void); + +extern const struct ltc_hash_descriptor sha224_desc; + #endif +#endif + +#ifdef LTC_SHA1 +int sha1_init(hash_state *md); +int sha1_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int sha1_done(hash_state *md, unsigned char *hash); +int sha1_test(void); + +extern const struct ltc_hash_descriptor sha1_desc; +#endif + +#ifdef LTC_MD5 +int md5_init(hash_state *md); +int md5_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int md5_done(hash_state *md, unsigned char *hash); +int md5_test(void); + +extern const struct ltc_hash_descriptor md5_desc; +#endif + +#ifdef LTC_MD4 +int md4_init(hash_state *md); +int md4_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int md4_done(hash_state *md, unsigned char *hash); +int md4_test(void); + +extern const struct ltc_hash_descriptor md4_desc; +#endif + +#ifdef LTC_MD2 +int md2_init(hash_state *md); +int md2_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int md2_done(hash_state *md, unsigned char *hash); +int md2_test(void); + +extern const struct ltc_hash_descriptor md2_desc; +#endif + +#ifdef LTC_TIGER +int tiger_init(hash_state *md); +int tiger_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int tiger_done(hash_state *md, unsigned char *hash); +int tiger_test(void); + +extern const struct ltc_hash_descriptor tiger_desc; +#endif + +#ifdef LTC_RIPEMD128 +int rmd128_init(hash_state *md); +int rmd128_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int rmd128_done(hash_state *md, unsigned char *hash); +int rmd128_test(void); + +extern const struct ltc_hash_descriptor rmd128_desc; +#endif + +#ifdef LTC_RIPEMD160 +int rmd160_init(hash_state *md); +int rmd160_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int rmd160_done(hash_state *md, unsigned char *hash); +int rmd160_test(void); + +extern const struct ltc_hash_descriptor rmd160_desc; +#endif + +#ifdef LTC_RIPEMD256 +int rmd256_init(hash_state *md); +int rmd256_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int rmd256_done(hash_state *md, unsigned char *hash); +int rmd256_test(void); + +extern const struct ltc_hash_descriptor rmd256_desc; +#endif + +#ifdef LTC_RIPEMD320 +int rmd320_init(hash_state *md); +int rmd320_process(hash_state *md, const unsigned char *in, unsigned long inlen); +int rmd320_done(hash_state *md, unsigned char *hash); +int rmd320_test(void); + +extern const struct ltc_hash_descriptor rmd320_desc; +#endif + + +int find_hash(const char *name); +int find_hash_id(unsigned char ID); +int find_hash_oid(const unsigned long *ID, unsigned long IDlen); +int find_hash_any(const char *name, int digestlen); +int register_hash(const struct ltc_hash_descriptor *hash); +int unregister_hash(const struct ltc_hash_descriptor *hash); +int hash_is_valid(int idx); + +LTC_MUTEX_PROTO(ltc_hash_mutex) + +int hash_memory(int hash, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int hash_memory_multi(int hash, unsigned char *out, unsigned long *outlen, + const unsigned char *in, unsigned long inlen, ...); +int hash_filehandle(int hash, FILE *in, unsigned char *out, unsigned long *outlen); +int hash_file(int hash, const char *fname, unsigned char *out, unsigned long *outlen); + +/* a simple macro for making hash "process" functions */ +#define HASH_PROCESS(func_name, compress_name, state_var, block_size) \ + int func_name(hash_state * md, const unsigned char *in, unsigned long inlen) \ + { \ + unsigned long n; \ + int err; \ + LTC_ARGCHK(md != NULL); \ + LTC_ARGCHK(in != NULL); \ + if (md->state_var.curlen > sizeof(md->state_var.buf)) { \ + return CRYPT_INVALID_ARG; \ + } \ + while (inlen > 0) { \ + if (md->state_var.curlen == 0 && inlen >= block_size) { \ + if ((err = compress_name(md, (unsigned char *)in)) != CRYPT_OK) { \ + return err; \ + } \ + md->state_var.length += block_size * 8; \ + in += block_size; \ + inlen -= block_size; \ + } else { \ + n = MIN(inlen, (block_size - md->state_var.curlen)); \ + memcpy(md->state_var.buf + md->state_var.curlen, in, (size_t)n); \ + md->state_var.curlen += n; \ + in += n; \ + inlen -= n; \ + if (md->state_var.curlen == block_size) { \ + if ((err = compress_name(md, md->state_var.buf)) != CRYPT_OK) { \ + return err; \ + } \ + md->state_var.length += 8 * block_size; \ + md->state_var.curlen = 0; \ + } \ + } \ + } \ + return CRYPT_OK; \ + } + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_hash.h,v $ */ +/* $Revision: 1.22 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + +#ifdef LTC_HMAC +typedef struct Hmac_state { + hash_state md; + int hash; + hash_state hashstate; + unsigned char *key; +} hmac_state; + +int hmac_init(hmac_state *hmac, int hash, const unsigned char *key, unsigned long keylen); +int hmac_process(hmac_state *hmac, const unsigned char *in, unsigned long inlen); +int hmac_done(hmac_state *hmac, unsigned char *out, unsigned long *outlen); +int hmac_test(void); +int hmac_memory(int hash, + const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int hmac_memory_multi(int hash, + const unsigned char *key, unsigned long keylen, + unsigned char *out, unsigned long *outlen, + const unsigned char *in, unsigned long inlen, ...); +int hmac_file(int hash, const char *fname, const unsigned char *key, + unsigned long keylen, + unsigned char *dst, unsigned long *dstlen); +#endif + +#ifdef LTC_OMAC + +typedef struct { + int cipher_idx, + buflen, + blklen; + unsigned char block[MAXBLOCKSIZE], + prev[MAXBLOCKSIZE], + Lu[2][MAXBLOCKSIZE]; + symmetric_key key; +} omac_state; + +int omac_init(omac_state *omac, int cipher, const unsigned char *key, unsigned long keylen); +int omac_process(omac_state *omac, const unsigned char *in, unsigned long inlen); +int omac_done(omac_state *omac, unsigned char *out, unsigned long *outlen); +int omac_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int omac_memory_multi(int cipher, + const unsigned char *key, unsigned long keylen, + unsigned char *out, unsigned long *outlen, + const unsigned char *in, unsigned long inlen, ...); +int omac_file(int cipher, + const unsigned char *key, unsigned long keylen, + const char *filename, + unsigned char *out, unsigned long *outlen); +int omac_test(void); +#endif /* LTC_OMAC */ + +#ifdef LTC_PMAC + +typedef struct { + unsigned char Ls[32][MAXBLOCKSIZE], /* L shifted by i bits to the left */ + Li[MAXBLOCKSIZE], /* value of Li [current value, we calc from previous recall] */ + Lr[MAXBLOCKSIZE], /* L * x^-1 */ + block[MAXBLOCKSIZE], /* currently accumulated block */ + checksum[MAXBLOCKSIZE]; /* current checksum */ + + symmetric_key key; /* scheduled key for cipher */ + unsigned long block_index; /* index # for current block */ + int cipher_idx, /* cipher idx */ + block_len, /* length of block */ + buflen; /* number of bytes in the buffer */ +} pmac_state; + +int pmac_init(pmac_state *pmac, int cipher, const unsigned char *key, unsigned long keylen); +int pmac_process(pmac_state *pmac, const unsigned char *in, unsigned long inlen); +int pmac_done(pmac_state *pmac, unsigned char *out, unsigned long *outlen); + +int pmac_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *msg, unsigned long msglen, + unsigned char *out, unsigned long *outlen); + +int pmac_memory_multi(int cipher, + const unsigned char *key, unsigned long keylen, + unsigned char *out, unsigned long *outlen, + const unsigned char *in, unsigned long inlen, ...); + +int pmac_file(int cipher, + const unsigned char *key, unsigned long keylen, + const char *filename, + unsigned char *out, unsigned long *outlen); + +int pmac_test(void); + +/* internal functions */ +int pmac_ntz(unsigned long x); +void pmac_shift_xor(pmac_state *pmac); +#endif /* PMAC */ + +#ifdef LTC_EAX_MODE + + #if !(defined(LTC_OMAC) && defined(LTC_CTR_MODE)) + #error LTC_EAX_MODE requires LTC_OMAC and CTR + #endif + +typedef struct { + unsigned char N[MAXBLOCKSIZE]; + symmetric_CTR ctr; + omac_state headeromac, ctomac; +} eax_state; + +int eax_init(eax_state *eax, int cipher, const unsigned char *key, unsigned long keylen, + const unsigned char *nonce, unsigned long noncelen, + const unsigned char *header, unsigned long headerlen); + +int eax_encrypt(eax_state *eax, const unsigned char *pt, unsigned char *ct, unsigned long length); +int eax_decrypt(eax_state *eax, const unsigned char *ct, unsigned char *pt, unsigned long length); +int eax_addheader(eax_state *eax, const unsigned char *header, unsigned long length); +int eax_done(eax_state *eax, unsigned char *tag, unsigned long *taglen); + +int eax_encrypt_authenticate_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *nonce, unsigned long noncelen, + const unsigned char *header, unsigned long headerlen, + const unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + unsigned char *tag, unsigned long *taglen); + +int eax_decrypt_verify_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *nonce, unsigned long noncelen, + const unsigned char *header, unsigned long headerlen, + const unsigned char *ct, unsigned long ctlen, + unsigned char *pt, + unsigned char *tag, unsigned long taglen, + int *stat); + +int eax_test(void); +#endif /* EAX MODE */ + +#ifdef LTC_OCB_MODE +typedef struct { + unsigned char L[MAXBLOCKSIZE], /* L value */ + Ls[32][MAXBLOCKSIZE], /* L shifted by i bits to the left */ + Li[MAXBLOCKSIZE], /* value of Li [current value, we calc from previous recall] */ + Lr[MAXBLOCKSIZE], /* L * x^-1 */ + R[MAXBLOCKSIZE], /* R value */ + checksum[MAXBLOCKSIZE]; /* current checksum */ + + symmetric_key key; /* scheduled key for cipher */ + unsigned long block_index; /* index # for current block */ + int cipher, /* cipher idx */ + block_len; /* length of block */ +} ocb_state; + +int ocb_init(ocb_state *ocb, int cipher, + const unsigned char *key, unsigned long keylen, const unsigned char *nonce); + +int ocb_encrypt(ocb_state *ocb, const unsigned char *pt, unsigned char *ct); +int ocb_decrypt(ocb_state *ocb, const unsigned char *ct, unsigned char *pt); + +int ocb_done_encrypt(ocb_state *ocb, + const unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + unsigned char *tag, unsigned long *taglen); + +int ocb_done_decrypt(ocb_state *ocb, + const unsigned char *ct, unsigned long ctlen, + unsigned char *pt, + const unsigned char *tag, unsigned long taglen, int *stat); + +int ocb_encrypt_authenticate_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *nonce, + const unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + unsigned char *tag, unsigned long *taglen); + +int ocb_decrypt_verify_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *nonce, + const unsigned char *ct, unsigned long ctlen, + unsigned char *pt, + const unsigned char *tag, unsigned long taglen, + int *stat); + +int ocb_test(void); + +/* internal functions */ +void ocb_shift_xor(ocb_state *ocb, unsigned char *Z); +int ocb_ntz(unsigned long x); +int s_ocb_done(ocb_state *ocb, const unsigned char *pt, unsigned long ptlen, + unsigned char *ct, unsigned char *tag, unsigned long *taglen, int mode); +#endif /* LTC_OCB_MODE */ + +#ifdef LTC_CCM_MODE + + #define CCM_ENCRYPT 0 + #define CCM_DECRYPT 1 + +int ccm_memory(int cipher, + const unsigned char *key, unsigned long keylen, + symmetric_key *uskey, + const unsigned char *nonce, unsigned long noncelen, + const unsigned char *header, unsigned long headerlen, + unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + unsigned char *tag, unsigned long *taglen, + int direction); + +int ccm_test(void); +#endif /* LTC_CCM_MODE */ + +#if defined(LRW_MODE) || defined(LTC_GCM_MODE) +void gcm_gf_mult(const unsigned char *a, const unsigned char *b, unsigned char *c); +#endif + + +/* table shared between GCM and LRW */ +#if defined(LTC_GCM_TABLES) || defined(LRW_TABLES) || ((defined(LTC_GCM_MODE) || defined(LTC_GCM_MODE)) && defined(LTC_FAST)) +extern const unsigned char gcm_shift_table[]; +#endif + +#ifdef LTC_GCM_MODE + + #define GCM_ENCRYPT 0 + #define GCM_DECRYPT 1 + + #define LTC_GCM_MODE_IV 0 + #define LTC_GCM_MODE_AAD 1 + #define LTC_GCM_MODE_TEXT 2 + +typedef struct { + symmetric_key K; + unsigned char H[16], /* multiplier */ + X[16], /* accumulator */ + Y[16], /* counter */ + Y_0[16], /* initial counter */ + buf[16]; /* buffer for stuff */ + + int cipher, /* which cipher */ + ivmode, /* Which mode is the IV in? */ + mode, /* mode the GCM code is in */ + buflen; /* length of data in buf */ + + ulong64 totlen, /* 64-bit counter used for IV and AAD */ + pttotlen; /* 64-bit counter for the PT */ + + #ifdef LTC_GCM_TABLES + unsigned char PC[16][256][16] /* 16 tables of 8x128 */ + #ifdef LTC_GCM_TABLES_SSE2 + __attribute__ ((aligned(16))) + #endif + ; + #endif +} gcm_state; + +void gcm_mult_h(gcm_state *gcm, unsigned char *I); + +int gcm_init(gcm_state *gcm, int cipher, + const unsigned char *key, int keylen); + +int gcm_reset(gcm_state *gcm); + +int gcm_add_iv(gcm_state *gcm, + const unsigned char *IV, unsigned long IVlen); + +int gcm_add_aad(gcm_state *gcm, + const unsigned char *adata, unsigned long adatalen); + +int gcm_process(gcm_state *gcm, + unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + int direction); + +int gcm_done(gcm_state *gcm, + unsigned char *tag, unsigned long *taglen); + +int gcm_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *IV, unsigned long IVlen, + const unsigned char *adata, unsigned long adatalen, + unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + unsigned char *tag, unsigned long *taglen, + int direction); +int gcm_test(void); +#endif /* LTC_GCM_MODE */ + +#ifdef LTC_PELICAN + +typedef struct pelican_state { + symmetric_key K; + unsigned char state[16]; + int buflen; +} pelican_state; + +int pelican_init(pelican_state *pelmac, const unsigned char *key, unsigned long keylen); +int pelican_process(pelican_state *pelmac, const unsigned char *in, unsigned long inlen); +int pelican_done(pelican_state *pelmac, unsigned char *out); +int pelican_test(void); + +int pelican_memory(const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out); +#endif + +#ifdef LTC_XCBC + +/* add this to "keylen" to xcbc_init to use a pure three-key XCBC MAC */ + #define LTC_XCBC_PURE 0x8000UL + +typedef struct { + unsigned char K[3][MAXBLOCKSIZE], + IV[MAXBLOCKSIZE]; + + symmetric_key key; + + int cipher, + buflen, + blocksize; +} xcbc_state; + +int xcbc_init(xcbc_state *xcbc, int cipher, const unsigned char *key, unsigned long keylen); +int xcbc_process(xcbc_state *xcbc, const unsigned char *in, unsigned long inlen); +int xcbc_done(xcbc_state *xcbc, unsigned char *out, unsigned long *outlen); +int xcbc_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int xcbc_memory_multi(int cipher, + const unsigned char *key, unsigned long keylen, + unsigned char *out, unsigned long *outlen, + const unsigned char *in, unsigned long inlen, ...); +int xcbc_file(int cipher, + const unsigned char *key, unsigned long keylen, + const char *filename, + unsigned char *out, unsigned long *outlen); +int xcbc_test(void); +#endif + +#ifdef LTC_F9_MODE + +typedef struct { + unsigned char akey[MAXBLOCKSIZE], + ACC[MAXBLOCKSIZE], + IV[MAXBLOCKSIZE]; + + symmetric_key key; + + int cipher, + buflen, + keylen, + blocksize; +} f9_state; + +int f9_init(f9_state *f9, int cipher, const unsigned char *key, unsigned long keylen); +int f9_process(f9_state *f9, const unsigned char *in, unsigned long inlen); +int f9_done(f9_state *f9, unsigned char *out, unsigned long *outlen); +int f9_memory(int cipher, + const unsigned char *key, unsigned long keylen, + const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int f9_memory_multi(int cipher, + const unsigned char *key, unsigned long keylen, + unsigned char *out, unsigned long *outlen, + const unsigned char *in, unsigned long inlen, ...); +int f9_file(int cipher, + const unsigned char *key, unsigned long keylen, + const char *filename, + unsigned char *out, unsigned long *outlen); +int f9_test(void); +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_mac.h,v $ */ +/* $Revision: 1.23 $ */ +/* $Date: 2007/05/12 14:37:41 $ */ + +/* ---- PRNG Stuff ---- */ +#ifdef LTC_YARROW +struct yarrow_prng { + int cipher, hash; + unsigned char pool[MAXBLOCKSIZE]; + symmetric_CTR ctr; + LTC_MUTEX_TYPE(prng_lock) +}; +#endif + +#ifdef LTC_RC4 +struct rc4_prng { + int x, y; + unsigned char buf[256]; +}; +#endif + +#ifdef LTC_FORTUNA +struct fortuna_prng { + hash_state pool[LTC_FORTUNA_POOLS]; /* the pools */ + + symmetric_key skey; + + unsigned char K[32], /* the current key */ + IV[16]; /* IV for CTR mode */ + + unsigned long pool_idx, /* current pool we will add to */ + pool0_len, /* length of 0'th pool */ + wd; + + ulong64 reset_cnt; /* number of times we have reset */ + LTC_MUTEX_TYPE(prng_lock) +}; +#endif + +#ifdef LTC_SOBER128 +struct sober128_prng { + ulong32 R[17], /* Working storage for the shift register */ + initR[17], /* saved register contents */ + konst, /* key dependent constant */ + sbuf; /* partial word encryption buffer */ + + int nbuf, /* number of part-word stream bits buffered */ + flag, /* first add_entropy call or not? */ + set; /* did we call add_entropy to set key? */ +}; +#endif + +typedef union Prng_state { + char dummy[1]; +#ifdef LTC_YARROW + struct yarrow_prng yarrow; +#endif +#ifdef LTC_RC4 + struct rc4_prng rc4; +#endif +#ifdef LTC_FORTUNA + struct fortuna_prng fortuna; +#endif +#ifdef LTC_SOBER128 + struct sober128_prng sober128; +#endif +} prng_state; + +/** PRNG descriptor */ +extern struct ltc_prng_descriptor { + /** Name of the PRNG */ + char *name; + /** size in bytes of exported state */ + int export_size; + + /** Start a PRNG state + @param prng [out] The state to initialize + @return CRYPT_OK if successful + */ + int (*start)(prng_state *prng); + + /** Add entropy to the PRNG + @param in The entropy + @param inlen Length of the entropy (octets)\ + @param prng The PRNG state + @return CRYPT_OK if successful + */ + int (*add_entropy)(const unsigned char *in, unsigned long inlen, prng_state *prng); + + /** Ready a PRNG state to read from + @param prng The PRNG state to ready + @return CRYPT_OK if successful + */ + int (*ready)(prng_state *prng); + + /** Read from the PRNG + @param out [out] Where to store the data + @param outlen Length of data desired (octets) + @param prng The PRNG state to read from + @return Number of octets read + */ + unsigned long (*read)(unsigned char *out, unsigned long outlen, prng_state *prng); + + /** Terminate a PRNG state + @param prng The PRNG state to terminate + @return CRYPT_OK if successful + */ + int (*done)(prng_state *prng); + + /** Export a PRNG state + @param out [out] The destination for the state + @param outlen [in/out] The max size and resulting size of the PRNG state + @param prng The PRNG to export + @return CRYPT_OK if successful + */ + int (*pexport)(unsigned char *out, unsigned long *outlen, prng_state *prng); + + /** Import a PRNG state + @param in The data to import + @param inlen The length of the data to import (octets) + @param prng The PRNG to initialize/import + @return CRYPT_OK if successful + */ + int (*pimport)(const unsigned char *in, unsigned long inlen, prng_state *prng); + + /** Self-test the PRNG + @return CRYPT_OK if successful, CRYPT_NOP if self-testing has been disabled + */ + int (*test)(void); +} prng_descriptor[]; + +#ifdef LTC_YARROW +int yarrow_start(prng_state *prng); +int yarrow_add_entropy(const unsigned char *in, unsigned long inlen, prng_state *prng); +int yarrow_ready(prng_state *prng); +unsigned long yarrow_read(unsigned char *out, unsigned long outlen, prng_state *prng); +int yarrow_done(prng_state *prng); +int yarrow_export(unsigned char *out, unsigned long *outlen, prng_state *prng); +int yarrow_import(const unsigned char *in, unsigned long inlen, prng_state *prng); +int yarrow_test(void); + +extern const struct ltc_prng_descriptor yarrow_desc; +#endif + +#ifdef LTC_FORTUNA +int fortuna_start(prng_state *prng); +int fortuna_add_entropy(const unsigned char *in, unsigned long inlen, prng_state *prng); +int fortuna_ready(prng_state *prng); +unsigned long fortuna_read(unsigned char *out, unsigned long outlen, prng_state *prng); +int fortuna_done(prng_state *prng); +int fortuna_export(unsigned char *out, unsigned long *outlen, prng_state *prng); +int fortuna_import(const unsigned char *in, unsigned long inlen, prng_state *prng); +int fortuna_test(void); + +extern const struct ltc_prng_descriptor fortuna_desc; +#endif + +#ifdef LTC_RC4 +int rc4_start(prng_state *prng); +int rc4_add_entropy(const unsigned char *in, unsigned long inlen, prng_state *prng); +int rc4_ready(prng_state *prng); +unsigned long rc4_read(unsigned char *out, unsigned long outlen, prng_state *prng); +int rc4_done(prng_state *prng); +int rc4_export(unsigned char *out, unsigned long *outlen, prng_state *prng); +int rc4_import(const unsigned char *in, unsigned long inlen, prng_state *prng); +int rc4_test(void); + +extern const struct ltc_prng_descriptor rc4_desc; +#endif + +#ifdef LTC_SPRNG +int sprng_start(prng_state *prng); +int sprng_add_entropy(const unsigned char *in, unsigned long inlen, prng_state *prng); +int sprng_ready(prng_state *prng); +unsigned long sprng_read(unsigned char *out, unsigned long outlen, prng_state *prng); +int sprng_done(prng_state *prng); +int sprng_export(unsigned char *out, unsigned long *outlen, prng_state *prng); +int sprng_import(const unsigned char *in, unsigned long inlen, prng_state *prng); +int sprng_test(void); + +extern const struct ltc_prng_descriptor sprng_desc; +#endif + +#ifdef LTC_SOBER128 +int sober128_start(prng_state *prng); +int sober128_add_entropy(const unsigned char *in, unsigned long inlen, prng_state *prng); +int sober128_ready(prng_state *prng); +unsigned long sober128_read(unsigned char *out, unsigned long outlen, prng_state *prng); +int sober128_done(prng_state *prng); +int sober128_export(unsigned char *out, unsigned long *outlen, prng_state *prng); +int sober128_import(const unsigned char *in, unsigned long inlen, prng_state *prng); +int sober128_test(void); + +extern const struct ltc_prng_descriptor sober128_desc; +#endif + +int find_prng(const char *name); +int register_prng(const struct ltc_prng_descriptor *prng); +int unregister_prng(const struct ltc_prng_descriptor *prng); +int prng_is_valid(int idx); + +LTC_MUTEX_PROTO(ltc_prng_mutex) + +/* Slow RNG you **might** be able to use to seed a PRNG with. Be careful as this + * might not work on all platforms as planned + */ +unsigned long rng_get_bytes(unsigned char *out, + unsigned long outlen, + void ( *callback)(void)); + +int rng_make_prng(int bits, int wprng, prng_state *prng, void (*callback)(void)); + + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_prng.h,v $ */ +/* $Revision: 1.9 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + +/* ---- NUMBER THEORY ---- */ + +enum { + PK_PUBLIC =0, + PK_PRIVATE=1 +}; + +int rand_prime(void *N, long len, prng_state *prng, int wprng); + +/* ---- RSA ---- */ +#ifdef LTC_MRSA + +/* Min and Max RSA key sizes (in bits) */ + #define MIN_RSA_SIZE 1024 + #define MAX_RSA_SIZE 4096 + +/** RSA LTC_PKCS style key */ +typedef struct Rsa_key { + /** Type of key, PK_PRIVATE or PK_PUBLIC */ + int type; + /** The public exponent */ + void *e; + /** The private exponent */ + void *d; + /** The modulus */ + void *N; + /** The p factor of N */ + void *p; + /** The q factor of N */ + void *q; + /** The 1/q mod p CRT param */ + void *qP; + /** The d mod (p - 1) CRT param */ + void *dP; + /** The d mod (q - 1) CRT param */ + void *dQ; +} rsa_key; + +int rsa_make_key(prng_state *prng, int wprng, int size, long e, rsa_key *key); + +int rsa_exptmod(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, int which, + rsa_key *key); + +void rsa_free(rsa_key *key); + +/* These use LTC_PKCS #1 v2.0 padding */ + #define rsa_encrypt_key(_in, _inlen, _out, _outlen, _lparam, _lparamlen, _prng, _prng_idx, _hash_idx, _key) \ + rsa_encrypt_key_ex(_in, _inlen, _out, _outlen, _lparam, _lparamlen, _prng, _prng_idx, _hash_idx, LTC_LTC_PKCS_1_OAEP, _key) + + #define rsa_decrypt_key(_in, _inlen, _out, _outlen, _lparam, _lparamlen, _hash_idx, _stat, _key) \ + rsa_decrypt_key_ex(_in, _inlen, _out, _outlen, _lparam, _lparamlen, _hash_idx, LTC_LTC_PKCS_1_OAEP, _stat, _key) + + #define rsa_sign_hash(_in, _inlen, _out, _outlen, _prng, _prng_idx, _hash_idx, _saltlen, _key) \ + rsa_sign_hash_ex(_in, _inlen, _out, _outlen, LTC_LTC_PKCS_1_PSS, _prng, _prng_idx, _hash_idx, _saltlen, _key) + + #define rsa_verify_hash(_sig, _siglen, _hash, _hashlen, _hash_idx, _saltlen, _stat, _key) \ + rsa_verify_hash_ex(_sig, _siglen, _hash, _hashlen, LTC_LTC_PKCS_1_PSS, _hash_idx, _saltlen, _stat, _key) + +/* These can be switched between LTC_PKCS #1 v2.x and LTC_PKCS #1 v1.5 paddings */ +int rsa_encrypt_key_ex(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + const unsigned char *lparam, unsigned long lparamlen, + prng_state *prng, int prng_idx, int hash_idx, int padding, rsa_key *key); + +int rsa_decrypt_key_ex(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + const unsigned char *lparam, unsigned long lparamlen, + int hash_idx, int padding, + int *stat, rsa_key *key); + +int rsa_sign_hash_ex(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + int padding, + prng_state *prng, int prng_idx, + int hash_idx, unsigned long saltlen, + rsa_key *key); + +int rsa_verify_hash_ex(const unsigned char *sig, unsigned long siglen, + const unsigned char *hash, unsigned long hashlen, + int padding, + int hash_idx, unsigned long saltlen, + int *stat, rsa_key *key); + +/* LTC_PKCS #1 import/export */ +int rsa_export(unsigned char *out, unsigned long *outlen, int type, rsa_key *key); +int rsa_import(const unsigned char *in, unsigned long inlen, rsa_key *key); +#endif + +/* ---- Katja ---- */ +#ifdef MKAT + +/* Min and Max KAT key sizes (in bits) */ + #define MIN_KAT_SIZE 1024 + #define MAX_KAT_SIZE 4096 + +/** Katja LTC_PKCS style key */ +typedef struct KAT_key { + /** Type of key, PK_PRIVATE or PK_PUBLIC */ + int type; + /** The private exponent */ + void *d; + /** The modulus */ + void *N; + /** The p factor of N */ + void *p; + /** The q factor of N */ + void *q; + /** The 1/q mod p CRT param */ + void *qP; + /** The d mod (p - 1) CRT param */ + void *dP; + /** The d mod (q - 1) CRT param */ + void *dQ; + /** The pq param */ + void *pq; +} katja_key; + +int katja_make_key(prng_state *prng, int wprng, int size, katja_key *key); + +int katja_exptmod(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, int which, + katja_key *key); + +void katja_free(katja_key *key); + +/* These use LTC_PKCS #1 v2.0 padding */ +int katja_encrypt_key(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + const unsigned char *lparam, unsigned long lparamlen, + prng_state *prng, int prng_idx, int hash_idx, katja_key *key); + +int katja_decrypt_key(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + const unsigned char *lparam, unsigned long lparamlen, + int hash_idx, int *stat, + katja_key *key); + +/* LTC_PKCS #1 import/export */ +int katja_export(unsigned char *out, unsigned long *outlen, int type, katja_key *key); +int katja_import(const unsigned char *in, unsigned long inlen, katja_key *key); +#endif + +/* ---- ECC Routines ---- */ +#ifdef LTC_MECC + +/* size of our temp buffers for exported keys */ + #define ECC_BUF_SIZE 256 + +/* max private key size */ + #define ECC_MAXSIZE 66 + +/** Structure defines a NIST GF(p) curve */ +typedef struct { + /** The size of the curve in octets */ + int size; + + /** name of curve */ + char *name; + + /** The prime that defines the field the curve is in (encoded in hex) */ + char *prime; + + /** The fields B param (hex) */ + char *B; + + /** The order of the curve (hex) */ + char *order; + + /** The x co-ordinate of the base point on the curve (hex) */ + char *Gx; + + /** The y co-ordinate of the base point on the curve (hex) */ + char *Gy; +} ltc_ecc_set_type; + +/** A point on a ECC curve, stored in Jacbobian format such that (x,y,z) => (x/z^2, y/z^3, 1) when interpretted as affine */ +typedef struct { + /** The x co-ordinate */ + void *x; + + /** The y co-ordinate */ + void *y; + + /** The z co-ordinate */ + void *z; +} ecc_point; + +/** An ECC key */ +typedef struct { + /** Type of key, PK_PRIVATE or PK_PUBLIC */ + int type; + + /** Index into the ltc_ecc_sets[] for the parameters of this curve; if -1, then this key is using user supplied curve in dp */ + int idx; + + /** pointer to domain parameters; either points to NIST curves (identified by idx >= 0) or user supplied curve */ + const ltc_ecc_set_type *dp; + + /** The public key */ + ecc_point pubkey; + + /** The private key */ + void *k; +} ecc_key; + +/** the ECC params provided */ +extern const ltc_ecc_set_type ltc_ecc_sets[]; + +int ecc_test(void); +void ecc_sizes(int *low, int *high); +int ecc_get_size(ecc_key *key); + +int ecc_make_key(prng_state *prng, int wprng, int keysize, ecc_key *key); +int ecc_make_key_ex(prng_state *prng, int wprng, ecc_key *key, const ltc_ecc_set_type *dp); +void ecc_free(ecc_key *key); + +int ecc_export(unsigned char *out, unsigned long *outlen, int type, ecc_key *key); +int ecc_import(const unsigned char *in, unsigned long inlen, ecc_key *key); +int ecc_import_ex(const unsigned char *in, unsigned long inlen, ecc_key *key, const ltc_ecc_set_type *dp); + +int ecc_ansi_x963_export(ecc_key *key, unsigned char *out, unsigned long *outlen); +int ecc_ansi_x963_import(const unsigned char *in, unsigned long inlen, ecc_key *key); +int ecc_ansi_x963_import_ex(const unsigned char *in, unsigned long inlen, ecc_key *key, ltc_ecc_set_type *dp); + +int ecc_shared_secret(ecc_key *private_key, ecc_key *public_key, + unsigned char *out, unsigned long *outlen); + +int ecc_encrypt_key(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + prng_state *prng, int wprng, int hash, + ecc_key *key); + +int ecc_decrypt_key(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + ecc_key *key); + +int ecc_sign_hash(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + prng_state *prng, int wprng, ecc_key *key); + +int ecc_verify_hash(const unsigned char *sig, unsigned long siglen, + const unsigned char *hash, unsigned long hashlen, + int *stat, ecc_key *key); + +/* low level functions */ +ecc_point *ltc_ecc_new_point(void); +void ltc_ecc_del_point(ecc_point *p); +int ltc_ecc_is_valid_idx(int n); + +/* point ops (mp == montgomery digit) */ + #if !defined(LTC_MECC_ACCEL) || defined(LTM_LTC_DESC) || defined(GMP_LTC_DESC) +/* R = 2P */ +int ltc_ecc_projective_dbl_point(ecc_point *P, ecc_point *R, void *modulus, void *mp); + +/* R = P + Q */ +int ltc_ecc_projective_add_point(ecc_point *P, ecc_point *Q, ecc_point *R, void *modulus, void *mp); + #endif + + #if defined(LTC_MECC_FP) +/* optimized point multiplication using fixed point cache (HAC algorithm 14.117) */ +int ltc_ecc_fp_mulmod(void *k, ecc_point *G, ecc_point *R, void *modulus, int map); + +/* functions for saving/loading/freeing/adding to fixed point cache */ +int ltc_ecc_fp_save_state(unsigned char **out, unsigned long *outlen); +int ltc_ecc_fp_restore_state(unsigned char *in, unsigned long inlen); +void ltc_ecc_fp_free(void); +int ltc_ecc_fp_add_point(ecc_point *g, void *modulus, int lock); + +/* lock/unlock all points currently in fixed point cache */ +void ltc_ecc_fp_tablelock(int lock); + #endif + +/* R = kG */ +int ltc_ecc_mulmod(void *k, ecc_point *G, ecc_point *R, void *modulus, int map); + + #ifdef LTC_ECC_SHAMIR +/* kA*A + kB*B = C */ +int ltc_ecc_mul2add(ecc_point *A, void *kA, + ecc_point *B, void *kB, + ecc_point *C, + void *modulus); + + #ifdef LTC_MECC_FP +/* Shamir's trick with optimized point multiplication using fixed point cache */ +int ltc_ecc_fp_mul2add(ecc_point *A, void *kA, + ecc_point *B, void *kB, + ecc_point *C, void *modulus); + #endif + #endif + + +/* map P to affine from projective */ +int ltc_ecc_map(ecc_point *P, void *modulus, void *mp); +#endif + +#ifdef LTC_MDSA + +/* Max diff between group and modulus size in bytes */ + #define LTC_MDSA_DELTA 512 + +/* Max DSA group size in bytes (default allows 4k-bit groups) */ + #define LTC_MDSA_MAX_GROUP 512 + +/** DSA key structure */ +typedef struct { + /** The key type, PK_PRIVATE or PK_PUBLIC */ + int type; + + /** The order of the sub-group used in octets */ + int qord; + + /** The generator */ + void *g; + + /** The prime used to generate the sub-group */ + void *q; + + /** The large prime that generats the field the contains the sub-group */ + void *p; + + /** The private key */ + void *x; + + /** The public key */ + void *y; +} dsa_key; + +int dsa_make_key(prng_state *prng, int wprng, int group_size, int modulus_size, dsa_key *key); +void dsa_free(dsa_key *key); + +int dsa_sign_hash_raw(const unsigned char *in, unsigned long inlen, + void *r, void *s, + prng_state *prng, int wprng, dsa_key *key); + +int dsa_sign_hash(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + prng_state *prng, int wprng, dsa_key *key); + +int dsa_verify_hash_raw(void *r, void *s, + const unsigned char *hash, unsigned long hashlen, + int *stat, dsa_key *key); + +int dsa_verify_hash(const unsigned char *sig, unsigned long siglen, + const unsigned char *hash, unsigned long hashlen, + int *stat, dsa_key *key); + +int dsa_encrypt_key(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + prng_state *prng, int wprng, int hash, + dsa_key *key); + +int dsa_decrypt_key(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + dsa_key *key); + +int dsa_import(const unsigned char *in, unsigned long inlen, dsa_key *key); +int dsa_export(unsigned char *out, unsigned long *outlen, int type, dsa_key *key); +int dsa_verify_key(dsa_key *key, int *stat); + +int dsa_shared_secret(void *private_key, void *base, + dsa_key *public_key, + unsigned char *out, unsigned long *outlen); +#endif + +#ifdef LTC_DER +/* DER handling */ + +enum { + LTC_ASN1_EOL, + LTC_ASN1_BOOLEAN, + LTC_ASN1_INTEGER, + LTC_ASN1_SHORT_INTEGER, + LTC_ASN1_BIT_STRING, + LTC_ASN1_OCTET_STRING, + LTC_ASN1_NULL, + LTC_ASN1_OBJECT_IDENTIFIER, + LTC_ASN1_IA5_STRING, + LTC_ASN1_PRINTABLE_STRING, + LTC_ASN1_UTF8_STRING, + LTC_ASN1_UTCTIME, + LTC_ASN1_CHOICE, + LTC_ASN1_SEQUENCE, + LTC_ASN1_SET, + LTC_ASN1_SETOF +}; + +/** A LTC ASN.1 list type */ +typedef struct ltc_asn1_list_ { + /** The LTC ASN.1 enumerated type identifier */ + int type; + /** The data to encode or place for decoding */ + void *data; + /** The size of the input or resulting output */ + unsigned long size; + /** The used flag, this is used by the CHOICE ASN.1 type to indicate which choice was made */ + int used; + /** prev/next entry in the list */ + struct ltc_asn1_list_ *prev, *next, *child, *parent; +} ltc_asn1_list; + + #define LTC_SET_ASN1(list, index, Type, Data, Size) \ + do { \ + int LTC_MACRO_temp = (index); \ + ltc_asn1_list *LTC_MACRO_list = (list); \ + LTC_MACRO_list[LTC_MACRO_temp].type = (Type); \ + LTC_MACRO_list[LTC_MACRO_temp].data = (void *)(Data); \ + LTC_MACRO_list[LTC_MACRO_temp].size = (Size); \ + LTC_MACRO_list[LTC_MACRO_temp].used = 0; \ + } while (0); + +/* SEQUENCE */ +int der_encode_sequence_ex(ltc_asn1_list *list, unsigned long inlen, + unsigned char *out, unsigned long *outlen, int type_of); + + #define der_encode_sequence(list, inlen, out, outlen) der_encode_sequence_ex(list, inlen, out, outlen, LTC_ASN1_SEQUENCE) + +int der_decode_sequence_ex(const unsigned char *in, unsigned long inlen, + ltc_asn1_list *list, unsigned long outlen, int ordered); + + #define der_decode_sequence(in, inlen, list, outlen) der_decode_sequence_ex(in, inlen, list, outlen, 1) + +int der_length_sequence(ltc_asn1_list *list, unsigned long inlen, + unsigned long *outlen); + +/* SET */ + #define der_decode_set(in, inlen, list, outlen) der_decode_sequence_ex(in, inlen, list, outlen, 0) + #define der_length_set der_length_sequence +int der_encode_set(ltc_asn1_list *list, unsigned long inlen, + unsigned char *out, unsigned long *outlen); + +int der_encode_setof(ltc_asn1_list *list, unsigned long inlen, + unsigned char *out, unsigned long *outlen); + +/* VA list handy helpers with triplets of */ +int der_encode_sequence_multi(unsigned char *out, unsigned long *outlen, ...); +int der_decode_sequence_multi(const unsigned char *in, unsigned long inlen, ...); + +/* FLEXI DECODER handle unknown list decoder */ +int der_decode_sequence_flexi(const unsigned char *in, unsigned long *inlen, ltc_asn1_list **out); +void der_free_sequence_flexi(ltc_asn1_list *list); +void der_sequence_free(ltc_asn1_list *in); + +/* BOOLEAN */ +int der_length_boolean(unsigned long *outlen); +int der_encode_boolean(int in, + unsigned char *out, unsigned long *outlen); +int der_decode_boolean(const unsigned char *in, unsigned long inlen, + int *out); + +/* INTEGER */ +int der_encode_integer(void *num, unsigned char *out, unsigned long *outlen); +int der_decode_integer(const unsigned char *in, unsigned long inlen, void *num); +int der_length_integer(void *num, unsigned long *len); + +/* INTEGER -- handy for 0..2^32-1 values */ +int der_decode_short_integer(const unsigned char *in, unsigned long inlen, unsigned long *num); +int der_encode_short_integer(unsigned long num, unsigned char *out, unsigned long *outlen); +int der_length_short_integer(unsigned long num, unsigned long *outlen); + +/* BIT STRING */ +int der_encode_bit_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int der_decode_bit_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int der_length_bit_string(unsigned long nbits, unsigned long *outlen); + +/* OCTET STRING */ +int der_encode_octet_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int der_decode_octet_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int der_length_octet_string(unsigned long noctets, unsigned long *outlen); + +/* OBJECT IDENTIFIER */ +int der_encode_object_identifier(unsigned long *words, unsigned long nwords, + unsigned char *out, unsigned long *outlen); +int der_decode_object_identifier(const unsigned char *in, unsigned long inlen, + unsigned long *words, unsigned long *outlen); +int der_length_object_identifier(unsigned long *words, unsigned long nwords, unsigned long *outlen); +unsigned long der_object_identifier_bits(unsigned long x); + +/* IA5 STRING */ +int der_encode_ia5_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int der_decode_ia5_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int der_length_ia5_string(const unsigned char *octets, unsigned long noctets, unsigned long *outlen); + +int der_ia5_char_encode(int c); +int der_ia5_value_decode(int v); + +/* Printable STRING */ +int der_encode_printable_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int der_decode_printable_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); +int der_length_printable_string(const unsigned char *octets, unsigned long noctets, unsigned long *outlen); + +int der_printable_char_encode(int c); +int der_printable_value_decode(int v); + +/* UTF-8 */ + #if (defined(SIZE_MAX) || __STDC_VERSION__ >= 199901L || defined(WCHAR_MAX) || defined(_WCHAR_T) || defined(_WCHAR_T_DEFINED) || defined (__WCHAR_TYPE__)) && !defined(LTC_NO_WCHAR) + #include + #else +typedef ulong32 wchar_t; + #endif + +int der_encode_utf8_string(const wchar_t *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen); + +int der_decode_utf8_string(const unsigned char *in, unsigned long inlen, + wchar_t *out, unsigned long *outlen); +unsigned long der_utf8_charsize(const wchar_t c); +int der_length_utf8_string(const wchar_t *in, unsigned long noctets, unsigned long *outlen); + + +/* CHOICE */ +int der_decode_choice(const unsigned char *in, unsigned long *inlen, + ltc_asn1_list *list, unsigned long outlen); + +/* UTCTime */ +typedef struct { + unsigned YY, /* year */ + MM, /* month */ + DD, /* day */ + hh, /* hour */ + mm, /* minute */ + ss, /* second */ + off_dir, /* timezone offset direction 0 == +, 1 == - */ + off_hh, /* timezone offset hours */ + off_mm; /* timezone offset minutes */ +} ltc_utctime; + +int der_encode_utctime(ltc_utctime *utctime, + unsigned char *out, unsigned long *outlen); + +int der_decode_utctime(const unsigned char *in, unsigned long *inlen, + ltc_utctime *out); + +int der_length_utctime(ltc_utctime *utctime, unsigned long *outlen); +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_pk.h,v $ */ +/* $Revision: 1.81 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + +/** math functions **/ +#define LTC_SOURCE +#define LTC_MP_LT -1 +#define LTC_MP_EQ 0 +#define LTC_MP_GT 1 + +#define LTC_MP_NO 0 +#define LTC_MP_YES 1 + +#ifndef LTC_MECC +typedef void ecc_point; +#endif + +#ifndef LTC_MRSA +typedef void rsa_key; +#endif + +/** math descriptor */ +typedef struct { + /** Name of the math provider */ + char *name; + + /** Bits per digit, amount of bits must fit in an unsigned long */ + int bits_per_digit; + +/* ---- init/deinit functions ---- */ + + /** initialize a bignum + @param a The number to initialize + @return CRYPT_OK on success + */ + int (*init)(void **a); + + /** init copy + @param dst The number to initialize and write to + @param src The number to copy from + @return CRYPT_OK on success + */ + int (*init_copy)(void **dst, void *src); + + /** deinit + @param a The number to free + @return CRYPT_OK on success + */ + void (*deinit)(void *a); + +/* ---- data movement ---- */ + + /** negate + @param src The number to negate + @param dst The destination + @return CRYPT_OK on success + */ + int (*neg)(void *src, void *dst); + + /** copy + @param src The number to copy from + @param dst The number to write to + @return CRYPT_OK on success + */ + int (*copy)(void *src, void *dst); + +/* ---- trivial low level functions ---- */ + + /** set small constant + @param a Number to write to + @param n Source upto bits_per_digit (actually meant for very small constants) + @return CRYPT_OK on succcess + */ + int (*set_int)(void *a, unsigned long n); + + /** get small constant + @param a Number to read, only fetches upto bits_per_digit from the number + @return The lower bits_per_digit of the integer (unsigned) + */ + unsigned long (*get_int)(void *a); + + /** get digit n + @param a The number to read from + @param n The number of the digit to fetch + @return The bits_per_digit sized n'th digit of a + */ + unsigned long (*get_digit)(void *a, int n); + + /** Get the number of digits that represent the number + @param a The number to count + @return The number of digits used to represent the number + */ + int (*get_digit_count)(void *a); + + /** compare two integers + @param a The left side integer + @param b The right side integer + @return LTC_MP_LT if a < b, LTC_MP_GT if a > b and LTC_MP_EQ otherwise. (signed comparison) + */ + int (*compare)(void *a, void *b); + + /** compare against int + @param a The left side integer + @param b The right side integer (upto bits_per_digit) + @return LTC_MP_LT if a < b, LTC_MP_GT if a > b and LTC_MP_EQ otherwise. (signed comparison) + */ + int (*compare_d)(void *a, unsigned long n); + + /** Count the number of bits used to represent the integer + @param a The integer to count + @return The number of bits required to represent the integer + */ + int (*count_bits)(void *a); + + /** Count the number of LSB bits which are zero + @param a The integer to count + @return The number of contiguous zero LSB bits + */ + int (*count_lsb_bits)(void *a); + + /** Compute a power of two + @param a The integer to store the power in + @param n The power of two you want to store (a = 2^n) + @return CRYPT_OK on success + */ + int (*twoexpt)(void *a, int n); + +/* ---- radix conversions ---- */ + + /** read ascii string + @param a The integer to store into + @param str The string to read + @param radix The radix the integer has been represented in (2-64) + @return CRYPT_OK on success + */ + int (*read_radix)(void *a, const char *str, int radix); + + /** write number to string + @param a The integer to store + @param str The destination for the string + @param radix The radix the integer is to be represented in (2-64) + @return CRYPT_OK on success + */ + int (*write_radix)(void *a, char *str, int radix); + + /** get size as unsigned char string + @param a The integer to get the size (when stored in array of octets) + @return The length of the integer + */ + unsigned long (*unsigned_size)(void *a); + + /** store an integer as an array of octets + @param src The integer to store + @param dst The buffer to store the integer in + @return CRYPT_OK on success + */ + int (*unsigned_write)(void *src, unsigned char *dst); + + /** read an array of octets and store as integer + @param dst The integer to load + @param src The array of octets + @param len The number of octets + @return CRYPT_OK on success + */ + int (*unsigned_read)(void *dst, unsigned char *src, unsigned long len); + +/* ---- basic math ---- */ + + /** add two integers + @param a The first source integer + @param b The second source integer + @param c The destination of "a + b" + @return CRYPT_OK on success + */ + int (*add)(void *a, void *b, void *c); + + + /** add two integers + @param a The first source integer + @param b The second source integer (single digit of upto bits_per_digit in length) + @param c The destination of "a + b" + @return CRYPT_OK on success + */ + int (*addi)(void *a, unsigned long b, void *c); + + /** subtract two integers + @param a The first source integer + @param b The second source integer + @param c The destination of "a - b" + @return CRYPT_OK on success + */ + int (*sub)(void *a, void *b, void *c); + + /** subtract two integers + @param a The first source integer + @param b The second source integer (single digit of upto bits_per_digit in length) + @param c The destination of "a - b" + @return CRYPT_OK on success + */ + int (*subi)(void *a, unsigned long b, void *c); + + /** multiply two integers + @param a The first source integer + @param b The second source integer (single digit of upto bits_per_digit in length) + @param c The destination of "a * b" + @return CRYPT_OK on success + */ + int (*mul)(void *a, void *b, void *c); + + /** multiply two integers + @param a The first source integer + @param b The second source integer (single digit of upto bits_per_digit in length) + @param c The destination of "a * b" + @return CRYPT_OK on success + */ + int (*muli)(void *a, unsigned long b, void *c); + + /** Square an integer + @param a The integer to square + @param b The destination + @return CRYPT_OK on success + */ + int (*sqr)(void *a, void *b); + + /** Divide an integer + @param a The dividend + @param b The divisor + @param c The quotient (can be NULL to signify don't care) + @param d The remainder (can be NULL to signify don't care) + @return CRYPT_OK on success + */ + int (*mpdiv)(void *a, void *b, void *c, void *d); + + /** divide by two + @param a The integer to divide (shift right) + @param b The destination + @return CRYPT_OK on success + */ + int (*div_2)(void *a, void *b); + + /** Get remainder (small value) + @param a The integer to reduce + @param b The modulus (upto bits_per_digit in length) + @param c The destination for the residue + @return CRYPT_OK on success + */ + int (*modi)(void *a, unsigned long b, unsigned long *c); + + /** gcd + @param a The first integer + @param b The second integer + @param c The destination for (a, b) + @return CRYPT_OK on success + */ + int (*gcd)(void *a, void *b, void *c); + + /** lcm + @param a The first integer + @param b The second integer + @param c The destination for [a, b] + @return CRYPT_OK on success + */ + int (*lcm)(void *a, void *b, void *c); + + /** Modular multiplication + @param a The first source + @param b The second source + @param c The modulus + @param d The destination (a*b mod c) + @return CRYPT_OK on success + */ + int (*mulmod)(void *a, void *b, void *c, void *d); + + /** Modular squaring + @param a The first source + @param b The modulus + @param c The destination (a*a mod b) + @return CRYPT_OK on success + */ + int (*sqrmod)(void *a, void *b, void *c); + + /** Modular inversion + @param a The value to invert + @param b The modulus + @param c The destination (1/a mod b) + @return CRYPT_OK on success + */ + int (*invmod)(void *, void *, void *); + +/* ---- reduction ---- */ + + /** setup montgomery + @param a The modulus + @param b The destination for the reduction digit + @return CRYPT_OK on success + */ + int (*montgomery_setup)(void *a, void **b); + + /** get normalization value + @param a The destination for the normalization value + @param b The modulus + @return CRYPT_OK on success + */ + int (*montgomery_normalization)(void *a, void *b); + + /** reduce a number + @param a The number [and dest] to reduce + @param b The modulus + @param c The value "b" from montgomery_setup() + @return CRYPT_OK on success + */ + int (*montgomery_reduce)(void *a, void *b, void *c); + + /** clean up (frees memory) + @param a The value "b" from montgomery_setup() + @return CRYPT_OK on success + */ + void (*montgomery_deinit)(void *a); + +/* ---- exponentiation ---- */ + + /** Modular exponentiation + @param a The base integer + @param b The power (can be negative) integer + @param c The modulus integer + @param d The destination + @return CRYPT_OK on success + */ + int (*exptmod)(void *a, void *b, void *c, void *d); + + /** Primality testing + @param a The integer to test + @param b The destination of the result (FP_YES if prime) + @return CRYPT_OK on success + */ + int (*isprime)(void *a, int *b); + +/* ---- (optional) ecc point math ---- */ + + /** ECC GF(p) point multiplication (from the NIST curves) + @param k The integer to multiply the point by + @param G The point to multiply + @param R The destination for kG + @param modulus The modulus for the field + @param map Boolean indicated whether to map back to affine or not (can be ignored if you work in affine only) + @return CRYPT_OK on success + */ + int (*ecc_ptmul)(void *k, ecc_point *G, ecc_point *R, void *modulus, int map); + + /** ECC GF(p) point addition + @param P The first point + @param Q The second point + @param R The destination of P + Q + @param modulus The modulus + @param mp The "b" value from montgomery_setup() + @return CRYPT_OK on success + */ + int (*ecc_ptadd)(ecc_point *P, ecc_point *Q, ecc_point *R, void *modulus, void *mp); + + /** ECC GF(p) point double + @param P The first point + @param R The destination of 2P + @param modulus The modulus + @param mp The "b" value from montgomery_setup() + @return CRYPT_OK on success + */ + int (*ecc_ptdbl)(ecc_point *P, ecc_point *R, void *modulus, void *mp); + + /** ECC mapping from projective to affine, currently uses (x,y,z) => (x/z^2, y/z^3, 1) + @param P The point to map + @param modulus The modulus + @param mp The "b" value from montgomery_setup() + @return CRYPT_OK on success + @remark The mapping can be different but keep in mind a ecc_point only has three + integers (x,y,z) so if you use a different mapping you have to make it fit. + */ + int (*ecc_map)(ecc_point *P, void *modulus, void *mp); + + /** Computes kA*A + kB*B = C using Shamir's Trick + @param A First point to multiply + @param kA What to multiple A by + @param B Second point to multiply + @param kB What to multiple B by + @param C [out] Destination point (can overlap with A or B + @param modulus Modulus for curve + @return CRYPT_OK on success + */ + int (*ecc_mul2add)(ecc_point *A, void *kA, + ecc_point *B, void *kB, + ecc_point *C, + void *modulus); + +/* ---- (optional) rsa optimized math (for internal CRT) ---- */ + + /** RSA Key Generation + @param prng An active PRNG state + @param wprng The index of the PRNG desired + @param size The size of the modulus (key size) desired (octets) + @param e The "e" value (public key). e==65537 is a good choice + @param key [out] Destination of a newly created private key pair + @return CRYPT_OK if successful, upon error all allocated ram is freed + */ + int (*rsa_keygen)(prng_state *prng, int wprng, int size, long e, rsa_key *key); + + + /** RSA exponentiation + @param in The octet array representing the base + @param inlen The length of the input + @param out The destination (to be stored in an octet array format) + @param outlen The length of the output buffer and the resulting size (zero padded to the size of the modulus) + @param which PK_PUBLIC for public RSA and PK_PRIVATE for private RSA + @param key The RSA key to use + @return CRYPT_OK on success + */ + int (*rsa_me)(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, int which, + rsa_key *key); +} ltc_math_descriptor; + +extern ltc_math_descriptor ltc_mp; + +int ltc_init_multi(void **a, ...); +void ltc_deinit_multi(void *a, ...); + +#ifdef LTM_DESC +extern const ltc_math_descriptor ltm_desc; +#endif + +#ifdef TFM_DESC +extern const ltc_math_descriptor tfm_desc; +#endif + +#ifdef GMP_DESC +extern const ltc_math_descriptor gmp_desc; +#endif + +#if !defined(DESC_DEF_ONLY) && defined(LTC_SOURCE) + #undef MP_DIGIT_BIT + #undef mp_iszero + #undef mp_isodd + #undef mp_tohex + + #define MP_DIGIT_BIT ltc_mp.bits_per_digit + +/* some handy macros */ + #define mp_init(a) ltc_mp.init(a) + #define mp_init_multi ltc_init_multi + #define mp_clear(a) ltc_mp.deinit(a) + #define mp_clear_multi ltc_deinit_multi + #define mp_init_copy(a, b) ltc_mp.init_copy(a, b) + + #define mp_neg(a, b) ltc_mp.neg(a, b) + #define mp_copy(a, b) ltc_mp.copy(a, b) + + #define mp_set(a, b) ltc_mp.set_int(a, b) + #define mp_set_int(a, b) ltc_mp.set_int(a, b) + #define mp_get_int(a) ltc_mp.get_int(a) + #define mp_get_digit(a, n) ltc_mp.get_digit(a, n) + #define mp_get_digit_count(a) ltc_mp.get_digit_count(a) + #define mp_cmp(a, b) ltc_mp.compare(a, b) + #define mp_cmp_d(a, b) ltc_mp.compare_d(a, b) + #define mp_count_bits(a) ltc_mp.count_bits(a) + #define mp_cnt_lsb(a) ltc_mp.count_lsb_bits(a) + #define mp_2expt(a, b) ltc_mp.twoexpt(a, b) + + #define mp_read_radix(a, b, c) ltc_mp.read_radix(a, b, c) + #define mp_toradix(a, b, c) ltc_mp.write_radix(a, b, c) + #define mp_unsigned_bin_size(a) ltc_mp.unsigned_size(a) + #define mp_to_unsigned_bin(a, b) ltc_mp.unsigned_write(a, b) + #define mp_read_unsigned_bin(a, b, c) ltc_mp.unsigned_read(a, b, c) + + #define mp_add(a, b, c) ltc_mp.add(a, b, c) + #define mp_add_d(a, b, c) ltc_mp.addi(a, b, c) + #define mp_sub(a, b, c) ltc_mp.sub(a, b, c) + #define mp_sub_d(a, b, c) ltc_mp.subi(a, b, c) + #define mp_mul(a, b, c) ltc_mp.mul(a, b, c) + #define mp_mul_d(a, b, c) ltc_mp.muli(a, b, c) + #define mp_sqr(a, b) ltc_mp.sqr(a, b) + #define mp_div(a, b, c, d) ltc_mp.mpdiv(a, b, c, d) + #define mp_div_2(a, b) ltc_mp.div_2(a, b) + #define mp_mod(a, b, c) ltc_mp.mpdiv(a, b, NULL, c) + #define mp_mod_d(a, b, c) ltc_mp.modi(a, b, c) + #define mp_gcd(a, b, c) ltc_mp.gcd(a, b, c) + #define mp_lcm(a, b, c) ltc_mp.lcm(a, b, c) + + #define mp_mulmod(a, b, c, d) ltc_mp.mulmod(a, b, c, d) + #define mp_sqrmod(a, b, c) ltc_mp.sqrmod(a, b, c) + #define mp_invmod(a, b, c) ltc_mp.invmod(a, b, c) + + #define mp_montgomery_setup(a, b) ltc_mp.montgomery_setup(a, b) + #define mp_montgomery_normalization(a, b) ltc_mp.montgomery_normalization(a, b) + #define mp_montgomery_reduce(a, b, c) ltc_mp.montgomery_reduce(a, b, c) + #define mp_montgomery_free(a) ltc_mp.montgomery_deinit(a) + + #define mp_exptmod(a, b, c, d) ltc_mp.exptmod(a, b, c, d) + #define mp_prime_is_prime(a, b, c) ltc_mp.isprime(a, c) + + #define mp_iszero(a) (mp_cmp_d(a, 0) == LTC_MP_EQ ? LTC_MP_YES : LTC_MP_NO) + #define mp_isodd(a) (mp_get_digit_count(a) > 0 ? (mp_get_digit(a, 0) & 1 ? LTC_MP_YES : LTC_MP_NO) : LTC_MP_NO) + #define mp_exch(a, b) do { void *ABC__tmp = a; a = b; b = ABC__tmp; } while (0); + + #define mp_tohex(a, b) mp_toradix(a, b, 16) +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_math.h,v $ */ +/* $Revision: 1.44 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + +/* ---- LTC_BASE64 Routines ---- */ +#ifdef LTC_BASE64 +int base64_encode(const unsigned char *in, unsigned long len, + unsigned char *out, unsigned long *outlen); + +int base64_decode(const unsigned char *in, unsigned long len, + unsigned char *out, unsigned long *outlen); +#endif + +/* ---- MEM routines ---- */ +void zeromem(void *dst, size_t len); +void burn_stack(unsigned long len); + +const char *error_to_string(int err); + +extern const char *crypt_build_settings; + +/* ---- HMM ---- */ +int crypt_fsa(void *mp, ...); + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_misc.h,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + +/* Defines the LTC_ARGCHK macro used within the library */ +/* ARGTYPE is defined in mycrypt_cfg.h */ +#if ARGTYPE == 0 + + #include + +/* this is the default LibTomCrypt macro */ +void crypt_argchk(char *v, char *s, int d); + + #define LTC_ARGCHK(x) if (!(x)) { crypt_argchk(#x, __FILE__, __LINE__); } + #define LTC_ARGCHKVD(x) LTC_ARGCHK(x) + +#elif ARGTYPE == 1 + +/* fatal type of error */ + #define LTC_ARGCHK(x) assert((x)) + #define LTC_ARGCHKVD(x) LTC_ARGCHK(x) + +#elif ARGTYPE == 2 + + #define LTC_ARGCHK(x) if (!(x)) { fprintf(stderr, "\nwarning: ARGCHK failed at %s:%d\n", __FILE__, __LINE__); } + #define LTC_ARGCHKVD(x) LTC_ARGCHK(x) + +#elif ARGTYPE == 3 + + #define LTC_ARGCHK(x) + #define LTC_ARGCHKVD(x) LTC_ARGCHK(x) + +#elif ARGTYPE == 4 + + #define LTC_ARGCHK(x) if (!(x)) return CRYPT_INVALID_ARG; + #define LTC_ARGCHKVD(x) if (!(x)) return; +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_argchk.h,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/08/27 20:50:21 $ */ + +/* LTC_PKCS Header Info */ + +/* ===> LTC_PKCS #1 -- RSA Cryptography <=== */ +#ifdef LTC_PKCS_1 + +enum ltc_pkcs_1_v1_5_blocks { + LTC_LTC_PKCS_1_EMSA = 1, /* Block type 1 (LTC_PKCS #1 v1.5 signature padding) */ + LTC_LTC_PKCS_1_EME = 2 /* Block type 2 (LTC_PKCS #1 v1.5 encryption padding) */ +}; + +enum ltc_pkcs_1_paddings { + LTC_LTC_PKCS_1_V1_5 = 1, /* LTC_PKCS #1 v1.5 padding (\sa ltc_pkcs_1_v1_5_blocks) */ + LTC_LTC_PKCS_1_OAEP = 2, /* LTC_PKCS #1 v2.0 encryption padding */ + LTC_LTC_PKCS_1_PSS = 3 /* LTC_PKCS #1 v2.1 signature padding */ +}; + +int pkcs_1_mgf1(int hash_idx, + const unsigned char *seed, unsigned long seedlen, + unsigned char *mask, unsigned long masklen); + +int pkcs_1_i2osp(void *n, unsigned long modulus_len, unsigned char *out); +int pkcs_1_os2ip(void *n, unsigned char *in, unsigned long inlen); + +/* *** v1.5 padding */ +int pkcs_1_v1_5_encode(const unsigned char *msg, + unsigned long msglen, + int block_type, + unsigned long modulus_bitlen, + prng_state *prng, + int prng_idx, + unsigned char *out, + unsigned long *outlen); + +int pkcs_1_v1_5_decode(const unsigned char *msg, + unsigned long msglen, + int block_type, + unsigned long modulus_bitlen, + unsigned char *out, + unsigned long *outlen, + int *is_valid); + +/* *** v2.1 padding */ +int pkcs_1_oaep_encode(const unsigned char *msg, unsigned long msglen, + const unsigned char *lparam, unsigned long lparamlen, + unsigned long modulus_bitlen, prng_state *prng, + int prng_idx, int hash_idx, + unsigned char *out, unsigned long *outlen); + +int pkcs_1_oaep_decode(const unsigned char *msg, unsigned long msglen, + const unsigned char *lparam, unsigned long lparamlen, + unsigned long modulus_bitlen, int hash_idx, + unsigned char *out, unsigned long *outlen, + int *res); + +int pkcs_1_pss_encode(const unsigned char *msghash, unsigned long msghashlen, + unsigned long saltlen, prng_state *prng, + int prng_idx, int hash_idx, + unsigned long modulus_bitlen, + unsigned char *out, unsigned long *outlen); + +int pkcs_1_pss_decode(const unsigned char *msghash, unsigned long msghashlen, + const unsigned char *sig, unsigned long siglen, + unsigned long saltlen, int hash_idx, + unsigned long modulus_bitlen, int *res); +#endif /* LTC_PKCS_1 */ + +/* ===> LTC_PKCS #5 -- Password Based Cryptography <=== */ +#ifdef LTC_PKCS_5 + +/* Algorithm #1 (old) */ +int pkcs_5_alg1(const unsigned char *password, unsigned long password_len, + const unsigned char *salt, + int iteration_count, int hash_idx, + unsigned char *out, unsigned long *outlen); + +/* Algorithm #2 (new) */ +int pkcs_5_alg2(const unsigned char *password, unsigned long password_len, + const unsigned char *salt, unsigned long salt_len, + int iteration_count, int hash_idx, + unsigned char *out, unsigned long *outlen); +#endif /* LTC_PKCS_5 */ + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt_pkcs.h,v $ */ +/* $Revision: 1.8 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + +#ifdef __cplusplus +} +#endif +#endif /* TOMCRYPT_H_ */ + + +/* $Source: /cvs/libtom/libtomcrypt/src/headers/tomcrypt.h,v $ */ +/* $Revision: 1.21 $ */ +/* $Date: 2006/12/16 19:34:05 $ */ + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_argchk.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_cipher_descriptor.c + Stores the cipher descriptor table, Tom St Denis + */ + +struct ltc_cipher_descriptor cipher_descriptor[TAB_SIZE] = { + { NULL, 0, 0, 0, 0, 0, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL } +}; + +LTC_MUTEX_GLOBAL(ltc_cipher_mutex) + + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_cipher_descriptor.c,v $ */ +/* $Revision: 1.13 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_cipher_is_valid.c + Determine if cipher is valid, Tom St Denis + */ + +/* + Test if a cipher index is valid + @param idx The index of the cipher to search for + @return CRYPT_OK if valid + */ +int cipher_is_valid(int idx) { + LTC_MUTEX_LOCK(<c_cipher_mutex); + if ((idx < 0) || (idx >= TAB_SIZE) || (cipher_descriptor[idx].name == NULL)) { + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return CRYPT_INVALID_CIPHER; + } + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return CRYPT_OK; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_cipher_is_valid.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_find_cipher.c + Find a cipher in the descriptor tables, Tom St Denis + */ + +/** + Find a registered cipher by name + @param name The name of the cipher to look for + @return >= 0 if found, -1 if not present + */ +int find_cipher(const char *name) { + int x; + + LTC_ARGCHK(name != NULL); + LTC_MUTEX_LOCK(<c_cipher_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if ((cipher_descriptor[x].name != NULL) && !XSTRCMP(cipher_descriptor[x].name, name)) { + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return x; + } + } + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_find_cipher.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_find_cipher_any.c + Find a cipher in the descriptor tables, Tom St Denis + */ + +/** + Find a cipher flexibly. First by name then if not present by block and key size + @param name The name of the cipher desired + @param blocklen The minimum length of the block cipher desired (octets) + @param keylen The minimum length of the key size desired (octets) + @return >= 0 if found, -1 if not present + */ +int find_cipher_any(const char *name, int blocklen, int keylen) { + int x; + + LTC_ARGCHK(name != NULL); + + x = find_cipher(name); + if (x != -1) return x; + + LTC_MUTEX_LOCK(<c_cipher_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if (cipher_descriptor[x].name == NULL) { + continue; + } + if ((blocklen <= (int)cipher_descriptor[x].block_length) && (keylen <= (int)cipher_descriptor[x].max_key_length)) { + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return x; + } + } + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_find_cipher_any.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_find_cipher_id.c + Find cipher by ID, Tom St Denis + */ + +/** + Find a cipher by ID number + @param ID The ID (not same as index) of the cipher to find + @return >= 0 if found, -1 if not present + */ +int find_cipher_id(unsigned char ID) { + int x; + + LTC_MUTEX_LOCK(<c_cipher_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if (cipher_descriptor[x].ID == ID) { + x = (cipher_descriptor[x].name == NULL) ? -1 : x; + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return x; + } + } + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_find_cipher_id.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_find_hash.c + Find a hash, Tom St Denis + */ + +/** + Find a registered hash by name + @param name The name of the hash to look for + @return >= 0 if found, -1 if not present + */ +int find_hash(const char *name) { + int x; + + LTC_ARGCHK(name != NULL); + LTC_MUTEX_LOCK(<c_hash_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if ((hash_descriptor[x].name != NULL) && (XSTRCMP(hash_descriptor[x].name, name) == 0)) { + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return x; + } + } + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_find_hash.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_find_hash_any.c + Find a hash, Tom St Denis + */ + +/** + Find a hash flexibly. First by name then if not present by digest size + @param name The name of the hash desired + @param digestlen The minimum length of the digest size (octets) + @return >= 0 if found, -1 if not present + */int find_hash_any(const char *name, int digestlen) { + int x, y, z; + + LTC_ARGCHK(name != NULL); + + x = find_hash(name); + if (x != -1) return x; + + LTC_MUTEX_LOCK(<c_hash_mutex); + y = MAXBLOCKSIZE + 1; + z = -1; + for (x = 0; x < TAB_SIZE; x++) { + if (hash_descriptor[x].name == NULL) { + continue; + } + if (((int)hash_descriptor[x].hashsize >= digestlen) && ((int)hash_descriptor[x].hashsize < y)) { + z = x; + y = hash_descriptor[x].hashsize; + } + } + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return z; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_find_hash_any.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_find_hash_id.c + Find hash by ID, Tom St Denis + */ + +/** + Find a hash by ID number + @param ID The ID (not same as index) of the hash to find + @return >= 0 if found, -1 if not present + */ +int find_hash_id(unsigned char ID) { + int x; + + LTC_MUTEX_LOCK(<c_hash_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if (hash_descriptor[x].ID == ID) { + x = (hash_descriptor[x].name == NULL) ? -1 : x; + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return x; + } + } + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_find_hash_id.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_find_hash_oid.c + Find a hash, Tom St Denis + */ + +int find_hash_oid(const unsigned long *ID, unsigned long IDlen) { + int x; + + LTC_ARGCHK(ID != NULL); + LTC_MUTEX_LOCK(<c_hash_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if ((hash_descriptor[x].name != NULL) && (hash_descriptor[x].OIDlen == IDlen) && !XMEMCMP(hash_descriptor[x].OID, ID, sizeof(unsigned long) * IDlen)) { + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return x; + } + } + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_find_hash_oid.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_find_prng.c + Find a PRNG, Tom St Denis + */ + +/** + Find a registered PRNG by name + @param name The name of the PRNG to look for + @return >= 0 if found, -1 if not present + */ +int find_prng(const char *name) { + int x; + + LTC_ARGCHK(name != NULL); + LTC_MUTEX_LOCK(<c_prng_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if ((prng_descriptor[x].name != NULL) && (XSTRCMP(prng_descriptor[x].name, name) == 0)) { + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return x; + } + } + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_find_prng.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +#include + +/** + @file crypt_fsa.c + LibTomCrypt FULL SPEED AHEAD!, Tom St Denis + */ + +/* format is ltc_mp, cipher_desc, [cipher_desc], NULL, hash_desc, [hash_desc], NULL, prng_desc, [prng_desc], NULL */ +int crypt_fsa(void *mp, ...) { + int err; + va_list args; + void *p; + + va_start(args, mp); + if (mp != NULL) { + XMEMCPY(<c_mp, mp, sizeof(ltc_mp)); + } + + while ((p = va_arg(args, void *)) != NULL) { + if ((err = register_cipher(p)) != CRYPT_OK) { + va_end(args); + return err; + } + } + + while ((p = va_arg(args, void *)) != NULL) { + if ((err = register_hash(p)) != CRYPT_OK) { + va_end(args); + return err; + } + } + + while ((p = va_arg(args, void *)) != NULL) { + if ((err = register_prng(p)) != CRYPT_OK) { + va_end(args); + return err; + } + } + + va_end(args); + return CRYPT_OK; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_fsa.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_hash_descriptor.c + Stores the hash descriptor table, Tom St Denis + */ + +struct ltc_hash_descriptor hash_descriptor[TAB_SIZE] = { + { NULL, 0, 0, 0, { 0 }, 0, NULL, NULL, NULL, NULL, NULL } +}; + +LTC_MUTEX_GLOBAL(ltc_hash_mutex) + + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_hash_descriptor.c,v $ */ +/* $Revision: 1.10 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_hash_is_valid.c + Determine if hash is valid, Tom St Denis + */ + +/* + Test if a hash index is valid + @param idx The index of the hash to search for + @return CRYPT_OK if valid + */ +int hash_is_valid(int idx) { + LTC_MUTEX_LOCK(<c_hash_mutex); + if ((idx < 0) || (idx >= TAB_SIZE) || (hash_descriptor[idx].name == NULL)) { + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return CRYPT_INVALID_HASH; + } + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return CRYPT_OK; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_hash_is_valid.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +ltc_math_descriptor ltc_mp; + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_prng_descriptor.c + Stores the PRNG descriptors, Tom St Denis + */ +struct ltc_prng_descriptor prng_descriptor[TAB_SIZE] = { + { NULL, 0, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL } +}; + +LTC_MUTEX_GLOBAL(ltc_prng_mutex) + + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_prng_descriptor.c,v $ */ +/* $Revision: 1.8 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_prng_is_valid.c + Determine if PRNG is valid, Tom St Denis + */ + +/* + Test if a PRNG index is valid + @param idx The index of the PRNG to search for + @return CRYPT_OK if valid + */ +int prng_is_valid(int idx) { + LTC_MUTEX_LOCK(<c_prng_mutex); + if ((idx < 0) || (idx >= TAB_SIZE) || (prng_descriptor[idx].name == NULL)) { + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return CRYPT_INVALID_PRNG; + } + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return CRYPT_OK; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_prng_is_valid.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_register_cipher.c + Register a cipher, Tom St Denis + */ + +/** + Register a cipher with the descriptor table + @param cipher The cipher you wish to register + @return value >= 0 if successfully added (or already present), -1 if unsuccessful + */ +int register_cipher(const struct ltc_cipher_descriptor *cipher) { + int x; + + LTC_ARGCHK(cipher != NULL); + + /* is it already registered? */ + LTC_MUTEX_LOCK(<c_cipher_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if ((cipher_descriptor[x].name != NULL) && (cipher_descriptor[x].ID == cipher->ID)) { + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return x; + } + } + + /* find a blank spot */ + for (x = 0; x < TAB_SIZE; x++) { + if (cipher_descriptor[x].name == NULL) { + XMEMCPY(&cipher_descriptor[x], cipher, sizeof(struct ltc_cipher_descriptor)); + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return x; + } + } + + /* no spot */ + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_register_cipher.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_register_hash.c + Register a HASH, Tom St Denis + */ + +/** + Register a hash with the descriptor table + @param hash The hash you wish to register + @return value >= 0 if successfully added (or already present), -1 if unsuccessful + */ +int register_hash(const struct ltc_hash_descriptor *hash) { + int x; + + LTC_ARGCHK(hash != NULL); + + /* is it already registered? */ + LTC_MUTEX_LOCK(<c_hash_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if (XMEMCMP(&hash_descriptor[x], hash, sizeof(struct ltc_hash_descriptor)) == 0) { + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return x; + } + } + + /* find a blank spot */ + for (x = 0; x < TAB_SIZE; x++) { + if (hash_descriptor[x].name == NULL) { + XMEMCPY(&hash_descriptor[x], hash, sizeof(struct ltc_hash_descriptor)); + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return x; + } + } + + /* no spot */ + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_register_hash.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_register_prng.c + Register a PRNG, Tom St Denis + */ + +/** + Register a PRNG with the descriptor table + @param prng The PRNG you wish to register + @return value >= 0 if successfully added (or already present), -1 if unsuccessful + */ +int register_prng(const struct ltc_prng_descriptor *prng) { + int x; + + LTC_ARGCHK(prng != NULL); + + /* is it already registered? */ + LTC_MUTEX_LOCK(<c_prng_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if (XMEMCMP(&prng_descriptor[x], prng, sizeof(struct ltc_prng_descriptor)) == 0) { + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return x; + } + } + + /* find a blank spot */ + for (x = 0; x < TAB_SIZE; x++) { + if (prng_descriptor[x].name == NULL) { + XMEMCPY(&prng_descriptor[x], prng, sizeof(struct ltc_prng_descriptor)); + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return x; + } + } + + /* no spot */ + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return -1; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_register_prng.c,v $ */ +/* $Revision: 1.8 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_unregister_cipher.c + Unregister a cipher, Tom St Denis + */ + +/** + Unregister a cipher from the descriptor table + @param cipher The cipher descriptor to remove + @return CRYPT_OK on success + */ +int unregister_cipher(const struct ltc_cipher_descriptor *cipher) { + int x; + + LTC_ARGCHK(cipher != NULL); + + /* is it already registered? */ + LTC_MUTEX_LOCK(<c_cipher_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if (XMEMCMP(&cipher_descriptor[x], cipher, sizeof(struct ltc_cipher_descriptor)) == 0) { + cipher_descriptor[x].name = NULL; + cipher_descriptor[x].ID = 255; + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return CRYPT_OK; + } + } + LTC_MUTEX_UNLOCK(<c_cipher_mutex); + return CRYPT_ERROR; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_unregister_cipher.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_unregister_hash.c + Unregister a hash, Tom St Denis + */ + +/** + Unregister a hash from the descriptor table + @param hash The hash descriptor to remove + @return CRYPT_OK on success + */ +int unregister_hash(const struct ltc_hash_descriptor *hash) { + int x; + + LTC_ARGCHK(hash != NULL); + + /* is it already registered? */ + LTC_MUTEX_LOCK(<c_hash_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if (XMEMCMP(&hash_descriptor[x], hash, sizeof(struct ltc_hash_descriptor)) == 0) { + hash_descriptor[x].name = NULL; + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return CRYPT_OK; + } + } + LTC_MUTEX_UNLOCK(<c_hash_mutex); + return CRYPT_ERROR; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_unregister_hash.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file crypt_unregister_prng.c + Unregister a PRNG, Tom St Denis + */ + +/** + Unregister a PRNG from the descriptor table + @param prng The PRNG descriptor to remove + @return CRYPT_OK on success + */ +int unregister_prng(const struct ltc_prng_descriptor *prng) { + int x; + + LTC_ARGCHK(prng != NULL); + + /* is it already registered? */ + LTC_MUTEX_LOCK(<c_prng_mutex); + for (x = 0; x < TAB_SIZE; x++) { + if (XMEMCMP(&prng_descriptor[x], prng, sizeof(struct ltc_prng_descriptor)) != 0) { + prng_descriptor[x].name = NULL; + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return CRYPT_OK; + } + } + LTC_MUTEX_UNLOCK(<c_prng_mutex); + return CRYPT_ERROR; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/crypt/crypt_unregister_prng.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_bit_string.c + ASN.1 DER, encode a BIT STRING, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store a BIT STRING + @param in The DER encoded BIT STRING + @param inlen The size of the DER BIT STRING + @param out [out] The array of bits stored (one per char) + @param outlen [in/out] The number of bits stored + @return CRYPT_OK if successful + */ +int der_decode_bit_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long dlen, blen, x, y; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* packet must be at least 4 bytes */ + if (inlen < 4) { + return CRYPT_INVALID_ARG; + } + + /* check for 0x03 */ + if ((in[0] & 0x1F) != 0x03) { + return CRYPT_INVALID_PACKET; + } + + /* offset in the data */ + x = 1; + + /* get the length of the data */ + if (in[x] & 0x80) { + /* long format get number of length bytes */ + y = in[x++] & 0x7F; + + /* invalid if 0 or > 2 */ + if ((y == 0) || (y > 2)) { + return CRYPT_INVALID_PACKET; + } + + /* read the data len */ + dlen = 0; + while (y--) { + dlen = (dlen << 8) | (unsigned long)in[x++]; + } + } else { + /* short format */ + dlen = in[x++] & 0x7F; + } + + /* is the data len too long or too short? */ + if ((dlen == 0) || (dlen + x > inlen)) { + return CRYPT_INVALID_PACKET; + } + + /* get padding count */ + blen = ((dlen - 1) << 3) - (in[x++] & 7); + + /* too many bits? */ + if (blen > *outlen) { + *outlen = blen; + return CRYPT_BUFFER_OVERFLOW; + } + + /* decode/store the bits */ + for (y = 0; y < blen; y++) { + out[y] = (in[x] & (1 << (7 - (y & 7)))) ? 1 : 0; + if ((y & 7) == 7) { + ++x; + } + } + + /* we done */ + *outlen = blen; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/bit/der_decode_bit_string.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_boolean.c + ASN.1 DER, decode a BOOLEAN, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Read a BOOLEAN + @param in The destination for the DER encoded BOOLEAN + @param inlen The size of the DER BOOLEAN + @param out [out] The boolean to decode + @return CRYPT_OK if successful + */ +int der_decode_boolean(const unsigned char *in, unsigned long inlen, + int *out) { + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + + if ((inlen != 3) || (in[0] != 0x01) || (in[1] != 0x01) || ((in[2] != 0x00) && (in[2] != 0xFF))) { + return CRYPT_INVALID_ARG; + } + + *out = (in[2] == 0xFF) ? 1 : 0; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/boolean/der_decode_boolean.c,v $ */ +/* $Revision: 1.2 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_choice.c + ASN.1 DER, decode a CHOICE, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Decode a CHOICE + @param in The DER encoded input + @param inlen [in/out] The size of the input and resulting size of read type + @param list The list of items to decode + @param outlen The number of items in the list + @return CRYPT_OK on success + */ +int der_decode_choice(const unsigned char *in, unsigned long *inlen, + ltc_asn1_list *list, unsigned long outlen) { + unsigned long size, x, z; + void *data; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(inlen != NULL); + LTC_ARGCHK(list != NULL); + + /* get blk size */ + if (*inlen < 2) { + return CRYPT_INVALID_PACKET; + } + + /* set all of the "used" flags to zero */ + for (x = 0; x < outlen; x++) { + list[x].used = 0; + } + + /* now scan until we have a winner */ + for (x = 0; x < outlen; x++) { + size = list[x].size; + data = list[x].data; + + switch (list[x].type) { + case LTC_ASN1_INTEGER: + if (der_decode_integer(in, *inlen, data) == CRYPT_OK) { + if (der_length_integer(data, &z) == CRYPT_OK) { + list[x].used = 1; + *inlen = z; + return CRYPT_OK; + } + } + break; + + case LTC_ASN1_SHORT_INTEGER: + if (der_decode_short_integer(in, *inlen, data) == CRYPT_OK) { + if (der_length_short_integer(size, &z) == CRYPT_OK) { + list[x].used = 1; + *inlen = z; + return CRYPT_OK; + } + } + break; + + case LTC_ASN1_BIT_STRING: + if (der_decode_bit_string(in, *inlen, data, &size) == CRYPT_OK) { + if (der_length_bit_string(size, &z) == CRYPT_OK) { + list[x].used = 1; + list[x].size = size; + *inlen = z; + return CRYPT_OK; + } + } + break; + + case LTC_ASN1_OCTET_STRING: + if (der_decode_octet_string(in, *inlen, data, &size) == CRYPT_OK) { + if (der_length_octet_string(size, &z) == CRYPT_OK) { + list[x].used = 1; + list[x].size = size; + *inlen = z; + return CRYPT_OK; + } + } + break; + + case LTC_ASN1_NULL: + if ((*inlen == 2) && (in[x] == 0x05) && (in[x + 1] == 0x00)) { + *inlen = 2; + list[x].used = 1; + return CRYPT_OK; + } + break; + + case LTC_ASN1_OBJECT_IDENTIFIER: + if (der_decode_object_identifier(in, *inlen, data, &size) == CRYPT_OK) { + if (der_length_object_identifier(data, size, &z) == CRYPT_OK) { + list[x].used = 1; + list[x].size = size; + *inlen = z; + return CRYPT_OK; + } + } + break; + + case LTC_ASN1_IA5_STRING: + if (der_decode_ia5_string(in, *inlen, data, &size) == CRYPT_OK) { + if (der_length_ia5_string(data, size, &z) == CRYPT_OK) { + list[x].used = 1; + list[x].size = size; + *inlen = z; + return CRYPT_OK; + } + } + break; + + + case LTC_ASN1_PRINTABLE_STRING: + if (der_decode_printable_string(in, *inlen, data, &size) == CRYPT_OK) { + if (der_length_printable_string(data, size, &z) == CRYPT_OK) { + list[x].used = 1; + list[x].size = size; + *inlen = z; + return CRYPT_OK; + } + } + break; + + case LTC_ASN1_UTF8_STRING: + if (der_decode_utf8_string(in, *inlen, data, &size) == CRYPT_OK) { + if (der_length_utf8_string(data, size, &z) == CRYPT_OK) { + list[x].used = 1; + list[x].size = size; + *inlen = z; + return CRYPT_OK; + } + } + break; + + case LTC_ASN1_UTCTIME: + z = *inlen; + if (der_decode_utctime(in, &z, data) == CRYPT_OK) { + list[x].used = 1; + *inlen = z; + return CRYPT_OK; + } + break; + + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + case LTC_ASN1_SEQUENCE: + if (der_decode_sequence(in, *inlen, data, size) == CRYPT_OK) { + if (der_length_sequence(data, size, &z) == CRYPT_OK) { + list[x].used = 1; + *inlen = z; + return CRYPT_OK; + } + } + break; + + default: + return CRYPT_INVALID_ARG; + } + } + + return CRYPT_INVALID_PACKET; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/choice/der_decode_choice.c,v $ */ +/* $Revision: 1.9 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_ia5_string.c + ASN.1 DER, encode a IA5 STRING, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store a IA5 STRING + @param in The DER encoded IA5 STRING + @param inlen The size of the DER IA5 STRING + @param out [out] The array of octets stored (one per char) + @param outlen [in/out] The number of octets stored + @return CRYPT_OK if successful + */ +int der_decode_ia5_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long x, y, len; + int t; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* must have header at least */ + if (inlen < 2) { + return CRYPT_INVALID_PACKET; + } + + /* check for 0x16 */ + if ((in[0] & 0x1F) != 0x16) { + return CRYPT_INVALID_PACKET; + } + x = 1; + + /* decode the length */ + if (in[x] & 0x80) { + /* valid # of bytes in length are 1,2,3 */ + y = in[x] & 0x7F; + if ((y == 0) || (y > 3) || ((x + y) > inlen)) { + return CRYPT_INVALID_PACKET; + } + + /* read the length in */ + len = 0; + ++x; + while (y--) { + len = (len << 8) | in[x++]; + } + } else { + len = in[x++] & 0x7F; + } + + /* is it too long? */ + if (len > *outlen) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + if (len + x > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* read the data */ + for (y = 0; y < len; y++) { + t = der_ia5_value_decode(in[x++]); + if (t == -1) { + return CRYPT_INVALID_ARG; + } + out[y] = t; + } + + *outlen = y; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/ia5/der_decode_ia5_string.c,v $ */ +/* $Revision: 1.4 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_integer.c + ASN.1 DER, decode an integer, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Read a mp_int integer + @param in The DER encoded data + @param inlen Size of DER encoded data + @param num The first mp_int to decode + @return CRYPT_OK if successful + */ +int der_decode_integer(const unsigned char *in, unsigned long inlen, void *num) { + unsigned long x, y, z; + int err; + + LTC_ARGCHK(num != NULL); + LTC_ARGCHK(in != NULL); + + /* min DER INTEGER is 0x02 01 00 == 0 */ + if (inlen < (1 + 1 + 1)) { + return CRYPT_INVALID_PACKET; + } + + /* ok expect 0x02 when we AND with 0001 1111 [1F] */ + x = 0; + if ((in[x++] & 0x1F) != 0x02) { + return CRYPT_INVALID_PACKET; + } + + /* now decode the len stuff */ + z = in[x++]; + + if ((z & 0x80) == 0x00) { + /* short form */ + + /* will it overflow? */ + if (x + z > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* no so read it */ + if ((err = mp_read_unsigned_bin(num, (unsigned char *)in + x, z)) != CRYPT_OK) { + return err; + } + } else { + /* long form */ + z &= 0x7F; + + /* will number of length bytes overflow? (or > 4) */ + if (((x + z) > inlen) || (z > 4) || (z == 0)) { + return CRYPT_INVALID_PACKET; + } + + /* now read it in */ + y = 0; + while (z--) { + y = ((unsigned long)(in[x++])) | (y << 8); + } + + /* now will reading y bytes overrun? */ + if ((x + y) > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* no so read it */ + if ((err = mp_read_unsigned_bin(num, (unsigned char *)in + x, y)) != CRYPT_OK) { + return err; + } + } + + /* see if it's negative */ + if (in[x] & 0x80) { + void *tmp; + if (mp_init(&tmp) != CRYPT_OK) { + return CRYPT_MEM; + } + + if ((mp_2expt(tmp, mp_count_bits(num)) != CRYPT_OK) || (mp_sub(num, tmp, num) != CRYPT_OK)) { + mp_clear(tmp); + return CRYPT_MEM; + } + mp_clear(tmp); + } + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/integer/der_decode_integer.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_object_identifier.c + ASN.1 DER, Decode Object Identifier, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Decode OID data and store the array of integers in words + @param in The OID DER encoded data + @param inlen The length of the OID data + @param words [out] The destination of the OID words + @param outlen [in/out] The number of OID words + @return CRYPT_OK if successful + */ +int der_decode_object_identifier(const unsigned char *in, unsigned long inlen, + unsigned long *words, unsigned long *outlen) { + unsigned long x, y, t, len; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(words != NULL); + LTC_ARGCHK(outlen != NULL); + + /* header is at least 3 bytes */ + if (inlen < 3) { + return CRYPT_INVALID_PACKET; + } + + /* must be room for at least two words */ + if (*outlen < 2) { + return CRYPT_BUFFER_OVERFLOW; + } + + /* decode the packet header */ + x = 0; + if ((in[x++] & 0x1F) != 0x06) { + return CRYPT_INVALID_PACKET; + } + + /* get the length */ + if (in[x] < 128) { + len = in[x++]; + } else { + if ((in[x] < 0x81) || (in[x] > 0x82)) { + return CRYPT_INVALID_PACKET; + } + y = in[x++] & 0x7F; + len = 0; + while (y--) { + len = (len << 8) | (unsigned long)in[x++]; + } + } + + if ((len < 1) || ((len + x) > inlen)) { + return CRYPT_INVALID_PACKET; + } + + /* decode words */ + y = 0; + t = 0; + while (len--) { + t = (t << 7) | (in[x] & 0x7F); + if (!(in[x++] & 0x80)) { + /* store t */ + if (y >= *outlen) { + return CRYPT_BUFFER_OVERFLOW; + } + if (y == 0) { + words[0] = t / 40; + words[1] = t % 40; + y = 2; + } else { + words[y++] = t; + } + t = 0; + } + } + + *outlen = y; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/object_identifier/der_decode_object_identifier.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_octet_string.c + ASN.1 DER, encode a OCTET STRING, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store a OCTET STRING + @param in The DER encoded OCTET STRING + @param inlen The size of the DER OCTET STRING + @param out [out] The array of octets stored (one per char) + @param outlen [in/out] The number of octets stored + @return CRYPT_OK if successful + */ +int der_decode_octet_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long x, y, len; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* must have header at least */ + if (inlen < 2) { + return CRYPT_INVALID_PACKET; + } + + /* check for 0x04 */ + if ((in[0] & 0x1F) != 0x04) { + return CRYPT_INVALID_PACKET; + } + x = 1; + + /* decode the length */ + if (in[x] & 0x80) { + /* valid # of bytes in length are 1,2,3 */ + y = in[x] & 0x7F; + if ((y == 0) || (y > 3) || ((x + y) > inlen)) { + return CRYPT_INVALID_PACKET; + } + + /* read the length in */ + len = 0; + ++x; + while (y--) { + len = (len << 8) | in[x++]; + } + } else { + len = in[x++] & 0x7F; + } + + /* is it too long? */ + if (len > *outlen) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + if (len + x > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* read the data */ + for (y = 0; y < len; y++) { + out[y] = in[x++]; + } + + *outlen = y; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/octet/der_decode_octet_string.c,v $ */ +/* $Revision: 1.4 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_printable_string.c + ASN.1 DER, encode a printable STRING, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store a printable STRING + @param in The DER encoded printable STRING + @param inlen The size of the DER printable STRING + @param out [out] The array of octets stored (one per char) + @param outlen [in/out] The number of octets stored + @return CRYPT_OK if successful + */ +int der_decode_printable_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long x, y, len; + int t; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* must have header at least */ + if (inlen < 2) { + return CRYPT_INVALID_PACKET; + } + + /* check for 0x13 */ + if ((in[0] & 0x1F) != 0x13) { + return CRYPT_INVALID_PACKET; + } + x = 1; + + /* decode the length */ + if (in[x] & 0x80) { + /* valid # of bytes in length are 1,2,3 */ + y = in[x] & 0x7F; + if ((y == 0) || (y > 3) || ((x + y) > inlen)) { + return CRYPT_INVALID_PACKET; + } + + /* read the length in */ + len = 0; + ++x; + while (y--) { + len = (len << 8) | in[x++]; + } + } else { + len = in[x++] & 0x7F; + } + + /* is it too long? */ + if (len > *outlen) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + if (len + x > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* read the data */ + for (y = 0; y < len; y++) { + t = der_printable_value_decode(in[x++]); + if (t == -1) { + return CRYPT_INVALID_ARG; + } + out[y] = t; + } + + *outlen = y; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/printable_string/der_decode_printable_string.c,v $ */ +/* $Revision: 1.4 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +#include + + +/** + @file der_decode_sequence_ex.c + ASN.1 DER, decode a SEQUENCE, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Decode a SEQUENCE + @param in The DER encoded input + @param inlen The size of the input + @param list The list of items to decode + @param outlen The number of items in the list + @param ordered Search an unordered or ordered list + @return CRYPT_OK on success + */ +int der_decode_sequence_ex(const unsigned char *in, unsigned long inlen, + ltc_asn1_list *list, unsigned long outlen, int ordered) { + int err, type; + unsigned long size, x, y, z, i, blksize; + void *data; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(list != NULL); + + /* get blk size */ + if (inlen < 2) { + return CRYPT_INVALID_PACKET; + } + + /* sequence type? We allow 0x30 SEQUENCE and 0x31 SET since fundamentally they're the same structure */ + x = 0; + if ((in[x] != 0x30) && (in[x] != 0x31)) { + return CRYPT_INVALID_PACKET; + } + ++x; + + if (in[x] < 128) { + blksize = in[x++]; + } else if (in[x] & 0x80) { + if ((in[x] < 0x81) || (in[x] > 0x83)) { + return CRYPT_INVALID_PACKET; + } + y = in[x++] & 0x7F; + + /* would reading the len bytes overrun? */ + if (x + y > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* read len */ + blksize = 0; + while (y--) { + blksize = (blksize << 8) | (unsigned long)in[x++]; + } + } + + /* would this blksize overflow? */ + if (x + blksize > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* mark all as unused */ + for (i = 0; i < outlen; i++) { + list[i].used = 0; + } + + /* ok read data */ + inlen = blksize; + for (i = 0; i < outlen; i++) { + z = 0; + type = list[i].type; + size = list[i].size; + data = list[i].data; + if (!ordered && (list[i].used == 1)) { + continue; + } + + if (type == LTC_ASN1_EOL) { + break; + } + + switch (type) { + case LTC_ASN1_BOOLEAN: + z = inlen; + if ((err = der_decode_boolean(in + x, z, ((int *)data))) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = der_length_boolean(&z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + case LTC_ASN1_INTEGER: + z = inlen; + if ((err = der_decode_integer(in + x, z, data)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + if ((err = der_length_integer(data, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + case LTC_ASN1_SHORT_INTEGER: + z = inlen; + if ((err = der_decode_short_integer(in + x, z, data)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + if ((err = der_length_short_integer(((unsigned long *)data)[0], &z)) != CRYPT_OK) { + goto LBL_ERR; + } + + break; + + case LTC_ASN1_BIT_STRING: + z = inlen; + if ((err = der_decode_bit_string(in + x, z, data, &size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + list[i].size = size; + if ((err = der_length_bit_string(size, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + case LTC_ASN1_OCTET_STRING: + z = inlen; + if ((err = der_decode_octet_string(in + x, z, data, &size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + list[i].size = size; + if ((err = der_length_octet_string(size, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + case LTC_ASN1_NULL: + if ((inlen < 2) || (in[x] != 0x05) || (in[x + 1] != 0x00)) { + if (!ordered) { + continue; + } + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + z = 2; + break; + + case LTC_ASN1_OBJECT_IDENTIFIER: + z = inlen; + if ((err = der_decode_object_identifier(in + x, z, data, &size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + list[i].size = size; + if ((err = der_length_object_identifier(data, size, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + case LTC_ASN1_IA5_STRING: + z = inlen; + if ((err = der_decode_ia5_string(in + x, z, data, &size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + list[i].size = size; + if ((err = der_length_ia5_string(data, size, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + + case LTC_ASN1_PRINTABLE_STRING: + z = inlen; + if ((err = der_decode_printable_string(in + x, z, data, &size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + list[i].size = size; + if ((err = der_length_printable_string(data, size, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + case LTC_ASN1_UTF8_STRING: + z = inlen; + if ((err = der_decode_utf8_string(in + x, z, data, &size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + list[i].size = size; + if ((err = der_length_utf8_string(data, size, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + case LTC_ASN1_UTCTIME: + z = inlen; + if ((err = der_decode_utctime(in + x, &z, data)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + break; + + case LTC_ASN1_SET: + z = inlen; + if ((err = der_decode_set(in + x, z, data, size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + if ((err = der_length_sequence(data, size, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + case LTC_ASN1_SETOF: + case LTC_ASN1_SEQUENCE: + /* detect if we have the right type */ + if (((type == LTC_ASN1_SETOF) && ((in[x] & 0x3F) != 0x31)) || ((type == LTC_ASN1_SEQUENCE) && ((in[x] & 0x3F) != 0x30))) { + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + + z = inlen; + if ((err = der_decode_sequence(in + x, z, data, size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + if ((err = der_length_sequence(data, size, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + break; + + + case LTC_ASN1_CHOICE: + z = inlen; + if ((err = der_decode_choice(in + x, &z, data, size)) != CRYPT_OK) { + if (!ordered) { + continue; + } + goto LBL_ERR; + } + break; + + default: + err = CRYPT_INVALID_ARG; + goto LBL_ERR; + } + x += z; + inlen -= z; + list[i].used = 1; + if (!ordered) { + /* restart the decoder */ + i = -1; + } + } + + for (i = 0; i < outlen; i++) { + if (list[i].used == 0) { + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + } + err = CRYPT_OK; + +LBL_ERR: + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/sequence/der_decode_sequence_ex.c,v $ */ +/* $Revision: 1.16 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_sequence_flexi.c + ASN.1 DER, decode an array of ASN.1 types with a flexi parser, Tom St Denis + */ + +#ifdef LTC_DER + +static unsigned long fetch_length(const unsigned char *in, unsigned long inlen) { + unsigned long x, y, z; + + y = 0; + + /* skip type and read len */ + if (inlen < 2) { + return 0xFFFFFFFF; + } + ++in; + ++y; + + /* read len */ + x = *in++; + ++y; + + /* <128 means literal */ + if (x < 128) { + return x + y; + } + x &= 0x7F; /* the lower 7 bits are the length of the length */ + inlen -= 2; + + /* len means len of len! */ + if ((x == 0) || (x > 4) || (x > inlen)) { + return 0xFFFFFFFF; + } + + y += x; + z = 0; + while (x--) { + z = (z << 8) | ((unsigned long)*in); + ++in; + } + return z + y; +} + +/** + ASN.1 DER Flexi(ble) decoder will decode arbitrary DER packets and create a linked list of the decoded elements. + @param in The input buffer + @param inlen [in/out] The length of the input buffer and on output the amount of decoded data + @param out [out] A pointer to the linked list + @return CRYPT_OK on success. + */ +int der_decode_sequence_flexi(const unsigned char *in, unsigned long *inlen, ltc_asn1_list **out) { + ltc_asn1_list *l; + unsigned long err, type, len, totlen, x, y; + void *realloc_tmp; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(inlen != NULL); + LTC_ARGCHK(out != NULL); + + l = NULL; + totlen = 0; + + /* scan the input and and get lengths and what not */ + while (*inlen) { + /* read the type byte */ + type = *in; + + /* fetch length */ + len = fetch_length(in, *inlen); + if (len > *inlen) { + err = CRYPT_INVALID_PACKET; + goto error; + } + + /* alloc new link */ + if (l == NULL) { + l = XCALLOC(1, sizeof(*l)); + if (l == NULL) { + err = CRYPT_MEM; + goto error; + } + } else { + l->next = XCALLOC(1, sizeof(*l)); + if (l->next == NULL) { + err = CRYPT_MEM; + goto error; + } + l->next->prev = l; + l = l->next; + } + + /* now switch on type */ + switch (type) { + case 0x01: /* BOOLEAN */ + l->type = LTC_ASN1_BOOLEAN; + l->size = 1; + l->data = XCALLOC(1, sizeof(int)); + + if ((err = der_decode_boolean(in, *inlen, l->data)) != CRYPT_OK) { + goto error; + } + + if ((err = der_length_boolean(&len)) != CRYPT_OK) { + goto error; + } + break; + + case 0x02: /* INTEGER */ + /* init field */ + l->type = LTC_ASN1_INTEGER; + l->size = 1; + if ((err = mp_init(&l->data)) != CRYPT_OK) { + goto error; + } + + /* decode field */ + if ((err = der_decode_integer(in, *inlen, l->data)) != CRYPT_OK) { + goto error; + } + + /* calc length of object */ + if ((err = der_length_integer(l->data, &len)) != CRYPT_OK) { + goto error; + } + break; + + case 0x03: /* BIT */ + /* init field */ + l->type = LTC_ASN1_BIT_STRING; + l->size = len * 8; /* *8 because we store decoded bits one per char and they are encoded 8 per char. */ + + if ((l->data = XCALLOC(1, l->size)) == NULL) { + err = CRYPT_MEM; + goto error; + } + + if ((err = der_decode_bit_string(in, *inlen, l->data, &l->size)) != CRYPT_OK) { + goto error; + } + + if ((err = der_length_bit_string(l->size, &len)) != CRYPT_OK) { + goto error; + } + break; + + case 0x04: /* OCTET */ + + /* init field */ + l->type = LTC_ASN1_OCTET_STRING; + l->size = len; + + if ((l->data = XCALLOC(1, l->size)) == NULL) { + err = CRYPT_MEM; + goto error; + } + + if ((err = der_decode_octet_string(in, *inlen, l->data, &l->size)) != CRYPT_OK) { + goto error; + } + + if ((err = der_length_octet_string(l->size, &len)) != CRYPT_OK) { + goto error; + } + break; + + case 0x05: /* NULL */ + + /* valid NULL is 0x05 0x00 */ + if ((in[0] != 0x05) || (in[1] != 0x00)) { + err = CRYPT_INVALID_PACKET; + goto error; + } + + /* simple to store ;-) */ + l->type = LTC_ASN1_NULL; + l->data = NULL; + l->size = 0; + len = 2; + + break; + + case 0x06: /* OID */ + + /* init field */ + l->type = LTC_ASN1_OBJECT_IDENTIFIER; + l->size = len; + + if ((l->data = XCALLOC(len, sizeof(unsigned long))) == NULL) { + err = CRYPT_MEM; + goto error; + } + + if ((err = der_decode_object_identifier(in, *inlen, l->data, &l->size)) != CRYPT_OK) { + goto error; + } + + if ((err = der_length_object_identifier(l->data, l->size, &len)) != CRYPT_OK) { + goto error; + } + + /* resize it to save a bunch of mem */ + if ((realloc_tmp = XREALLOC(l->data, l->size * sizeof(unsigned long))) == NULL) { + /* out of heap but this is not an error */ + break; + } + l->data = realloc_tmp; + break; + + case 0x0C: /* UTF8 */ + + /* init field */ + l->type = LTC_ASN1_UTF8_STRING; + l->size = len; + + if ((l->data = XCALLOC(l->size, sizeof(wchar_t))) == NULL) { + err = CRYPT_MEM; + goto error; + } + + if ((err = der_decode_utf8_string(in, *inlen, l->data, &l->size)) != CRYPT_OK) { + goto error; + } + + if ((err = der_length_utf8_string(l->data, l->size, &len)) != CRYPT_OK) { + goto error; + } + break; + + case 0x13: /* PRINTABLE */ + + /* init field */ + l->type = LTC_ASN1_PRINTABLE_STRING; + l->size = len; + + if ((l->data = XCALLOC(1, l->size)) == NULL) { + err = CRYPT_MEM; + goto error; + } + + if ((err = der_decode_printable_string(in, *inlen, l->data, &l->size)) != CRYPT_OK) { + goto error; + } + + if ((err = der_length_printable_string(l->data, l->size, &len)) != CRYPT_OK) { + goto error; + } + break; + + case 0x16: /* IA5 */ + + /* init field */ + l->type = LTC_ASN1_IA5_STRING; + l->size = len; + + if ((l->data = XCALLOC(1, l->size)) == NULL) { + err = CRYPT_MEM; + goto error; + } + + if ((err = der_decode_ia5_string(in, *inlen, l->data, &l->size)) != CRYPT_OK) { + goto error; + } + + if ((err = der_length_ia5_string(l->data, l->size, &len)) != CRYPT_OK) { + goto error; + } + break; + + case 0x17: /* UTC TIME */ + + /* init field */ + l->type = LTC_ASN1_UTCTIME; + l->size = 1; + + if ((l->data = XCALLOC(1, sizeof(ltc_utctime))) == NULL) { + err = CRYPT_MEM; + goto error; + } + + len = *inlen; + if ((err = der_decode_utctime(in, &len, l->data)) != CRYPT_OK) { + goto error; + } + + if ((err = der_length_utctime(l->data, &len)) != CRYPT_OK) { + goto error; + } + break; + + case 0x30: /* SEQUENCE */ + case 0x31: /* SET */ + + /* init field */ + l->type = (type == 0x30) ? LTC_ASN1_SEQUENCE : LTC_ASN1_SET; + + /* we have to decode the SEQUENCE header and get it's length */ + + /* move past type */ + ++in; + --(*inlen); + + /* read length byte */ + x = *in++; + --(*inlen); + + /* smallest SEQUENCE/SET header */ + y = 2; + + /* now if it's > 127 the next bytes are the length of the length */ + if (x > 128) { + x &= 0x7F; + in += x; + *inlen -= x; + + /* update sequence header len */ + y += x; + } + + /* Sequence elements go as child */ + len = len - y; + if ((err = der_decode_sequence_flexi(in, &len, &(l->child))) != CRYPT_OK) { + goto error; + } + + /* len update */ + totlen += y; + + /* link them up y0 */ + l->child->parent = l; + + break; + + default: + /* invalid byte ... this is a soft error */ + /* remove link */ + l = l->prev; + XFREE(l->next); + l->next = NULL; + goto outside; + } + + /* advance pointers */ + totlen += len; + in += len; + *inlen -= len; + } + +outside: + + /* rewind l please */ + while (l->prev != NULL || l->parent != NULL) { + if (l->parent != NULL) { + l = l->parent; + } else { + l = l->prev; + } + } + + /* return */ + *out = l; + *inlen = totlen; + return CRYPT_OK; + +error: + /* free list */ + der_sequence_free(l); + + return err; +} +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/sequence/der_decode_sequence_flexi.c,v $ */ +/* $Revision: 1.26 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +#include + + +/** + @file der_decode_sequence_multi.c + ASN.1 DER, decode a SEQUENCE, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Decode a SEQUENCE type using a VA list + @param in Input buffer + @param inlen Length of input in octets + @remark <...> is of the form (int, unsigned long, void*) + @return CRYPT_OK on success + */ +int der_decode_sequence_multi(const unsigned char *in, unsigned long inlen, ...) { + int err, type; + unsigned long size, x; + void *data; + va_list args; + ltc_asn1_list *list; + + LTC_ARGCHK(in != NULL); + + /* get size of output that will be required */ + va_start(args, inlen); + x = 0; + for ( ; ; ) { + type = va_arg(args, int); + size = va_arg(args, unsigned long); + data = va_arg(args, void *); + + if (type == LTC_ASN1_EOL) { + break; + } + + switch (type) { + case LTC_ASN1_BOOLEAN: + case LTC_ASN1_INTEGER: + case LTC_ASN1_SHORT_INTEGER: + case LTC_ASN1_BIT_STRING: + case LTC_ASN1_OCTET_STRING: + case LTC_ASN1_NULL: + case LTC_ASN1_OBJECT_IDENTIFIER: + case LTC_ASN1_IA5_STRING: + case LTC_ASN1_PRINTABLE_STRING: + case LTC_ASN1_UTF8_STRING: + case LTC_ASN1_UTCTIME: + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + case LTC_ASN1_SEQUENCE: + case LTC_ASN1_CHOICE: + ++x; + break; + + default: + va_end(args); + return CRYPT_INVALID_ARG; + } + } + va_end(args); + + /* allocate structure for x elements */ + if (x == 0) { + return CRYPT_NOP; + } + + list = XCALLOC(x, sizeof(*list)); + if (list == NULL) { + return CRYPT_MEM; + } + + /* fill in the structure */ + va_start(args, inlen); + x = 0; + for ( ; ; ) { + type = va_arg(args, int); + size = va_arg(args, unsigned long); + data = va_arg(args, void *); + + if (type == LTC_ASN1_EOL) { + break; + } + + switch (type) { + case LTC_ASN1_BOOLEAN: + case LTC_ASN1_INTEGER: + case LTC_ASN1_SHORT_INTEGER: + case LTC_ASN1_BIT_STRING: + case LTC_ASN1_OCTET_STRING: + case LTC_ASN1_NULL: + case LTC_ASN1_OBJECT_IDENTIFIER: + case LTC_ASN1_IA5_STRING: + case LTC_ASN1_PRINTABLE_STRING: + case LTC_ASN1_UTF8_STRING: + case LTC_ASN1_UTCTIME: + case LTC_ASN1_SEQUENCE: + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + case LTC_ASN1_CHOICE: + list[x].type = type; + list[x].size = size; + list[x++].data = data; + break; + + default: + va_end(args); + err = CRYPT_INVALID_ARG; + goto LBL_ERR; + } + } + va_end(args); + + err = der_decode_sequence(in, inlen, list, x); +LBL_ERR: + XFREE(list); + return err; +} +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/sequence/der_decode_sequence_multi.c,v $ */ +/* $Revision: 1.13 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_short_integer.c + ASN.1 DER, decode an integer, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Read a short integer + @param in The DER encoded data + @param inlen Size of data + @param num [out] The integer to decode + @return CRYPT_OK if successful + */ +int der_decode_short_integer(const unsigned char *in, unsigned long inlen, unsigned long *num) { + unsigned long len, x, y; + + LTC_ARGCHK(num != NULL); + LTC_ARGCHK(in != NULL); + + /* check length */ + if (inlen < 2) { + return CRYPT_INVALID_PACKET; + } + + /* check header */ + x = 0; + if ((in[x++] & 0x1F) != 0x02) { + return CRYPT_INVALID_PACKET; + } + + /* get the packet len */ + len = in[x++]; + + if (x + len > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* read number */ + y = 0; + while (len--) { + y = (y << 8) | (unsigned long)in[x++]; + } + *num = y; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/short_integer/der_decode_short_integer.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_utctime.c + ASN.1 DER, decode a UTCTIME, Tom St Denis + */ + +#ifdef LTC_DER + +static int char_to_int(unsigned char x) { + switch (x) { + case '0': + return 0; + + case '1': + return 1; + + case '2': + return 2; + + case '3': + return 3; + + case '4': + return 4; + + case '5': + return 5; + + case '6': + return 6; + + case '7': + return 7; + + case '8': + return 8; + + case '9': + return 9; + } + return 100; +} + + #define DECODE_V(y, max) \ + y = char_to_int(buf[x]) * 10 + char_to_int(buf[x + 1]); \ + if (y >= max) return CRYPT_INVALID_PACKET; \ + x += 2; + +/** + Decodes a UTC time structure in DER format (reads all 6 valid encoding formats) + @param in Input buffer + @param inlen Length of input buffer in octets + @param out [out] Destination of UTC time structure + @return CRYPT_OK if successful + */ +int der_decode_utctime(const unsigned char *in, unsigned long *inlen, + ltc_utctime *out) { + unsigned char buf[32]; + unsigned long x; + int y; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(inlen != NULL); + LTC_ARGCHK(out != NULL); + + /* check header */ + if ((*inlen < 2UL) || (in[1] >= sizeof(buf)) || ((in[1] + 2UL) > *inlen)) { + return CRYPT_INVALID_PACKET; + } + + /* decode the string */ + for (x = 0; x < in[1]; x++) { + y = der_ia5_value_decode(in[x + 2]); + if (y == -1) { + return CRYPT_INVALID_PACKET; + } + buf[x] = y; + } + *inlen = 2 + x; + + + /* possible encodings are + YYMMDDhhmmZ + YYMMDDhhmm+hh'mm' + YYMMDDhhmm-hh'mm' + YYMMDDhhmmssZ + YYMMDDhhmmss+hh'mm' + YYMMDDhhmmss-hh'mm' + + So let's do a trivial decode upto [including] mm + */ + + x = 0; + DECODE_V(out->YY, 100); + DECODE_V(out->MM, 13); + DECODE_V(out->DD, 32); + DECODE_V(out->hh, 24); + DECODE_V(out->mm, 60); + + /* clear timezone and seconds info */ + out->off_dir = out->off_hh = out->off_mm = out->ss = 0; + + /* now is it Z, +, - or 0-9 */ + if (buf[x] == 'Z') { + return CRYPT_OK; + } else if ((buf[x] == '+') || (buf[x] == '-')) { + out->off_dir = (buf[x++] == '+') ? 0 : 1; + DECODE_V(out->off_hh, 24); + DECODE_V(out->off_mm, 60); + return CRYPT_OK; + } + + /* decode seconds */ + DECODE_V(out->ss, 60); + + /* now is it Z, +, - */ + if (buf[x] == 'Z') { + return CRYPT_OK; + } else if ((buf[x] == '+') || (buf[x] == '-')) { + out->off_dir = (buf[x++] == '+') ? 0 : 1; + DECODE_V(out->off_hh, 24); + DECODE_V(out->off_mm, 60); + return CRYPT_OK; + } else { + return CRYPT_INVALID_PACKET; + } +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/utctime/der_decode_utctime.c,v $ */ +/* $Revision: 1.9 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_decode_utf8_string.c + ASN.1 DER, encode a UTF8 STRING, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store a UTF8 STRING + @param in The DER encoded UTF8 STRING + @param inlen The size of the DER UTF8 STRING + @param out [out] The array of utf8s stored (one per char) + @param outlen [in/out] The number of utf8s stored + @return CRYPT_OK if successful + */ +int der_decode_utf8_string(const unsigned char *in, unsigned long inlen, + wchar_t *out, unsigned long *outlen) { + wchar_t tmp; + unsigned long x, y, z, len; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* must have header at least */ + if (inlen < 2) { + return CRYPT_INVALID_PACKET; + } + + /* check for 0x0C */ + if ((in[0] & 0x1F) != 0x0C) { + return CRYPT_INVALID_PACKET; + } + x = 1; + + /* decode the length */ + if (in[x] & 0x80) { + /* valid # of bytes in length are 1,2,3 */ + y = in[x] & 0x7F; + if ((y == 0) || (y > 3) || ((x + y) > inlen)) { + return CRYPT_INVALID_PACKET; + } + + /* read the length in */ + len = 0; + ++x; + while (y--) { + len = (len << 8) | in[x++]; + } + } else { + len = in[x++] & 0x7F; + } + + if (len + x > inlen) { + return CRYPT_INVALID_PACKET; + } + + /* proceed to decode */ + for (y = 0; x < inlen; ) { + /* get first byte */ + tmp = in[x++]; + + /* count number of bytes */ + for (z = 0; (tmp & 0x80) && (z <= 4); z++, tmp = (tmp << 1) & 0xFF); + + if ((z > 4) || (x + (z - 1) > inlen)) { + return CRYPT_INVALID_PACKET; + } + + /* decode, grab upper bits */ + tmp >>= z; + + /* grab remaining bytes */ + if (z > 1) { + --z; + } + while (z-- != 0) { + if ((in[x] & 0xC0) != 0x80) { + return CRYPT_INVALID_PACKET; + } + tmp = (tmp << 6) | ((wchar_t)in[x++] & 0x3F); + } + + if (y > *outlen) { + *outlen = y; + return CRYPT_BUFFER_OVERFLOW; + } + out[y++] = tmp; + } + *outlen = y; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/utf8/der_decode_utf8_string.c,v $ */ +/* $Revision: 1.8 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_bit_string.c + ASN.1 DER, encode a BIT STRING, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store a BIT STRING + @param in The array of bits to store (one per char) + @param inlen The number of bits tostore + @param out [out] The destination for the DER encoded BIT STRING + @param outlen [in/out] The max size and resulting size of the DER BIT STRING + @return CRYPT_OK if successful + */ +int der_encode_bit_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long len, x, y; + unsigned char buf; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* avoid overflows */ + if ((err = der_length_bit_string(inlen, &len)) != CRYPT_OK) { + return err; + } + + if (len > *outlen) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + /* store header (include bit padding count in length) */ + x = 0; + y = (inlen >> 3) + ((inlen & 7) ? 1 : 0) + 1; + + out[x++] = 0x03; + if (y < 128) { + out[x++] = (unsigned char)y; + } else if (y < 256) { + out[x++] = 0x81; + out[x++] = (unsigned char)y; + } else if (y < 65536) { + out[x++] = 0x82; + out[x++] = (unsigned char)((y >> 8) & 255); + out[x++] = (unsigned char)(y & 255); + } + + /* store number of zero padding bits */ + out[x++] = (unsigned char)((8 - inlen) & 7); + + /* store the bits in big endian format */ + for (y = buf = 0; y < inlen; y++) { + buf |= (in[y] ? 1 : 0) << (7 - (y & 7)); + if ((y & 7) == 7) { + out[x++] = buf; + buf = 0; + } + } + /* store last byte */ + if (inlen & 7) { + out[x++] = buf; + } + *outlen = x; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/bit/der_encode_bit_string.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_boolean.c + ASN.1 DER, encode a BOOLEAN, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store a BOOLEAN + @param in The boolean to encode + @param out [out] The destination for the DER encoded BOOLEAN + @param outlen [in/out] The max size and resulting size of the DER BOOLEAN + @return CRYPT_OK if successful + */ +int der_encode_boolean(int in, + unsigned char *out, unsigned long *outlen) { + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(out != NULL); + + if (*outlen < 3) { + *outlen = 3; + return CRYPT_BUFFER_OVERFLOW; + } + + *outlen = 3; + out[0] = 0x01; + out[1] = 0x01; + out[2] = in ? 0xFF : 0x00; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/boolean/der_encode_boolean.c,v $ */ +/* $Revision: 1.4 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_ia5_string.c + ASN.1 DER, encode a IA5 STRING, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Store an IA5 STRING + @param in The array of IA5 to store (one per char) + @param inlen The number of IA5 to store + @param out [out] The destination for the DER encoded IA5 STRING + @param outlen [in/out] The max size and resulting size of the DER IA5 STRING + @return CRYPT_OK if successful + */ +int der_encode_ia5_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long x, y, len; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* get the size */ + if ((err = der_length_ia5_string(in, inlen, &len)) != CRYPT_OK) { + return err; + } + + /* too big? */ + if (len > *outlen) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + /* encode the header+len */ + x = 0; + out[x++] = 0x16; + if (inlen < 128) { + out[x++] = (unsigned char)inlen; + } else if (inlen < 256) { + out[x++] = 0x81; + out[x++] = (unsigned char)inlen; + } else if (inlen < 65536UL) { + out[x++] = 0x82; + out[x++] = (unsigned char)((inlen >> 8) & 255); + out[x++] = (unsigned char)(inlen & 255); + } else if (inlen < 16777216UL) { + out[x++] = 0x83; + out[x++] = (unsigned char)((inlen >> 16) & 255); + out[x++] = (unsigned char)((inlen >> 8) & 255); + out[x++] = (unsigned char)(inlen & 255); + } else { + return CRYPT_INVALID_ARG; + } + + /* store octets */ + for (y = 0; y < inlen; y++) { + out[x++] = der_ia5_char_encode(in[y]); + } + + /* retun length */ + *outlen = x; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/ia5/der_encode_ia5_string.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_integer.c + ASN.1 DER, encode an integer, Tom St Denis + */ + + +#ifdef LTC_DER + +/* Exports a positive bignum as DER format (upto 2^32 bytes in size) */ + +/** + Store a mp_int integer + @param num The first mp_int to encode + @param out [out] The destination for the DER encoded integers + @param outlen [in/out] The max size and resulting size of the DER encoded integers + @return CRYPT_OK if successful + */ +int der_encode_integer(void *num, unsigned char *out, unsigned long *outlen) { + unsigned long tmplen, y; + int err, leading_zero; + + LTC_ARGCHK(num != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* find out how big this will be */ + if ((err = der_length_integer(num, &tmplen)) != CRYPT_OK) { + return err; + } + + if (*outlen < tmplen) { + *outlen = tmplen; + return CRYPT_BUFFER_OVERFLOW; + } + + if (mp_cmp_d(num, 0) != LTC_MP_LT) { + /* we only need a leading zero if the msb of the first byte is one */ + if (((mp_count_bits(num) & 7) == 0) || (mp_iszero(num) == LTC_MP_YES)) { + leading_zero = 1; + } else { + leading_zero = 0; + } + + /* get length of num in bytes (plus 1 since we force the msbyte to zero) */ + y = mp_unsigned_bin_size(num) + leading_zero; + } else { + leading_zero = 0; + y = mp_count_bits(num); + y = y + (8 - (y & 7)); + y = y >> 3; + if (((mp_cnt_lsb(num) + 1) == mp_count_bits(num)) && ((mp_count_bits(num) & 7) == 0)) --y; + } + + /* now store initial data */ + *out++ = 0x02; + if (y < 128) { + /* short form */ + *out++ = (unsigned char)y; + } else if (y < 256) { + *out++ = 0x81; + *out++ = (unsigned char)y; + } else if (y < 65536UL) { + *out++ = 0x82; + *out++ = (unsigned char)((y >> 8) & 255); + *out++ = (unsigned char)y; + } else if (y < 16777216UL) { + *out++ = 0x83; + *out++ = (unsigned char)((y >> 16) & 255); + *out++ = (unsigned char)((y >> 8) & 255); + *out++ = (unsigned char)y; + } else { + return CRYPT_INVALID_ARG; + } + + /* now store msbyte of zero if num is non-zero */ + if (leading_zero) { + *out++ = 0x00; + } + + /* if it's not zero store it as big endian */ + if (mp_cmp_d(num, 0) == LTC_MP_GT) { + /* now store the mpint */ + if ((err = mp_to_unsigned_bin(num, out)) != CRYPT_OK) { + return err; + } + } else if (mp_iszero(num) != LTC_MP_YES) { + void *tmp; + + /* negative */ + if (mp_init(&tmp) != CRYPT_OK) { + return CRYPT_MEM; + } + + /* 2^roundup and subtract */ + y = mp_count_bits(num); + y = y + (8 - (y & 7)); + if (((mp_cnt_lsb(num) + 1) == mp_count_bits(num)) && ((mp_count_bits(num) & 7) == 0)) y -= 8; + if ((mp_2expt(tmp, y) != CRYPT_OK) || (mp_add(tmp, num, tmp) != CRYPT_OK)) { + mp_clear(tmp); + return CRYPT_MEM; + } + if ((err = mp_to_unsigned_bin(tmp, out)) != CRYPT_OK) { + mp_clear(tmp); + return err; + } + mp_clear(tmp); + } + + /* we good */ + *outlen = tmplen; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/integer/der_encode_integer.c,v $ */ +/* $Revision: 1.9 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_object_identifier.c + ASN.1 DER, Encode Object Identifier, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Encode an OID + @param words The words to encode (upto 32-bits each) + @param nwords The number of words in the OID + @param out [out] Destination of OID data + @param outlen [in/out] The max and resulting size of the OID + @return CRYPT_OK if successful + */ +int der_encode_object_identifier(unsigned long *words, unsigned long nwords, + unsigned char *out, unsigned long *outlen) { + unsigned long i, x, y, z, t, mask, wordbuf; + int err; + + LTC_ARGCHK(words != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* check length */ + if ((err = der_length_object_identifier(words, nwords, &x)) != CRYPT_OK) { + return err; + } + if (x > *outlen) { + *outlen = x; + return CRYPT_BUFFER_OVERFLOW; + } + + /* compute length to store OID data */ + z = 0; + wordbuf = words[0] * 40 + words[1]; + for (y = 1; y < nwords; y++) { + t = der_object_identifier_bits(wordbuf); + z += t / 7 + ((t % 7) ? 1 : 0) + (wordbuf == 0 ? 1 : 0); + if (y < nwords - 1) { + wordbuf = words[y + 1]; + } + } + + /* store header + length */ + x = 0; + out[x++] = 0x06; + if (z < 128) { + out[x++] = (unsigned char)z; + } else if (z < 256) { + out[x++] = 0x81; + out[x++] = (unsigned char)z; + } else if (z < 65536UL) { + out[x++] = 0x82; + out[x++] = (unsigned char)((z >> 8) & 255); + out[x++] = (unsigned char)(z & 255); + } else { + return CRYPT_INVALID_ARG; + } + + /* store first byte */ + wordbuf = words[0] * 40 + words[1]; + for (i = 1; i < nwords; i++) { + /* store 7 bit words in little endian */ + t = wordbuf & 0xFFFFFFFF; + if (t) { + y = x; + mask = 0; + while (t) { + out[x++] = (unsigned char)((t & 0x7F) | mask); + t >>= 7; + mask |= 0x80; /* upper bit is set on all but the last byte */ + } + /* now swap bytes y...x-1 */ + z = x - 1; + while (y < z) { + t = out[y]; + out[y] = out[z]; + out[z] = (unsigned char)t; + ++y; + --z; + } + } else { + /* zero word */ + out[x++] = 0x00; + } + + if (i < nwords - 1) { + wordbuf = words[i + 1]; + } + } + + *outlen = x; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/object_identifier/der_encode_object_identifier.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_octet_string.c + ASN.1 DER, encode a OCTET STRING, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store an OCTET STRING + @param in The array of OCTETS to store (one per char) + @param inlen The number of OCTETS to store + @param out [out] The destination for the DER encoded OCTET STRING + @param outlen [in/out] The max size and resulting size of the DER OCTET STRING + @return CRYPT_OK if successful + */ +int der_encode_octet_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long x, y, len; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* get the size */ + if ((err = der_length_octet_string(inlen, &len)) != CRYPT_OK) { + return err; + } + + /* too big? */ + if (len > *outlen) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + /* encode the header+len */ + x = 0; + out[x++] = 0x04; + if (inlen < 128) { + out[x++] = (unsigned char)inlen; + } else if (inlen < 256) { + out[x++] = 0x81; + out[x++] = (unsigned char)inlen; + } else if (inlen < 65536UL) { + out[x++] = 0x82; + out[x++] = (unsigned char)((inlen >> 8) & 255); + out[x++] = (unsigned char)(inlen & 255); + } else if (inlen < 16777216UL) { + out[x++] = 0x83; + out[x++] = (unsigned char)((inlen >> 16) & 255); + out[x++] = (unsigned char)((inlen >> 8) & 255); + out[x++] = (unsigned char)(inlen & 255); + } else { + return CRYPT_INVALID_ARG; + } + + /* store octets */ + for (y = 0; y < inlen; y++) { + out[x++] = in[y]; + } + + /* retun length */ + *outlen = x; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/octet/der_encode_octet_string.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_printable_string.c + ASN.1 DER, encode a printable STRING, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Store an printable STRING + @param in The array of printable to store (one per char) + @param inlen The number of printable to store + @param out [out] The destination for the DER encoded printable STRING + @param outlen [in/out] The max size and resulting size of the DER printable STRING + @return CRYPT_OK if successful + */ +int der_encode_printable_string(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long x, y, len; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* get the size */ + if ((err = der_length_printable_string(in, inlen, &len)) != CRYPT_OK) { + return err; + } + + /* too big? */ + if (len > *outlen) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + /* encode the header+len */ + x = 0; + out[x++] = 0x13; + if (inlen < 128) { + out[x++] = (unsigned char)inlen; + } else if (inlen < 256) { + out[x++] = 0x81; + out[x++] = (unsigned char)inlen; + } else if (inlen < 65536UL) { + out[x++] = 0x82; + out[x++] = (unsigned char)((inlen >> 8) & 255); + out[x++] = (unsigned char)(inlen & 255); + } else if (inlen < 16777216UL) { + out[x++] = 0x83; + out[x++] = (unsigned char)((inlen >> 16) & 255); + out[x++] = (unsigned char)((inlen >> 8) & 255); + out[x++] = (unsigned char)(inlen & 255); + } else { + return CRYPT_INVALID_ARG; + } + + /* store octets */ + for (y = 0; y < inlen; y++) { + out[x++] = der_printable_char_encode(in[y]); + } + + /* retun length */ + *outlen = x; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/printable_string/der_encode_printable_string.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +#include + + +/** + @file der_encode_sequence_ex.c + ASN.1 DER, encode a SEQUENCE, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Encode a SEQUENCE + @param list The list of items to encode + @param inlen The number of items in the list + @param out [out] The destination + @param outlen [in/out] The size of the output + @param type_of LTC_ASN1_SEQUENCE or LTC_ASN1_SET/LTC_ASN1_SETOF + @return CRYPT_OK on success + */ +int der_encode_sequence_ex(ltc_asn1_list *list, unsigned long inlen, + unsigned char *out, unsigned long *outlen, int type_of) { + int err, type; + unsigned long size, x, y, z, i; + void *data; + + LTC_ARGCHK(list != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* get size of output that will be required */ + y = 0; + for (i = 0; i < inlen; i++) { + type = list[i].type; + size = list[i].size; + data = list[i].data; + + if (type == LTC_ASN1_EOL) { + break; + } + + switch (type) { + case LTC_ASN1_BOOLEAN: + if ((err = der_length_boolean(&x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_INTEGER: + if ((err = der_length_integer(data, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_SHORT_INTEGER: + if ((err = der_length_short_integer(*((unsigned long *)data), &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_BIT_STRING: + if ((err = der_length_bit_string(size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_OCTET_STRING: + if ((err = der_length_octet_string(size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_NULL: + y += 2; + break; + + case LTC_ASN1_OBJECT_IDENTIFIER: + if ((err = der_length_object_identifier(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_IA5_STRING: + if ((err = der_length_ia5_string(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_PRINTABLE_STRING: + if ((err = der_length_printable_string(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_UTF8_STRING: + if ((err = der_length_utf8_string(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_UTCTIME: + if ((err = der_length_utctime(data, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + case LTC_ASN1_SEQUENCE: + if ((err = der_length_sequence(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + default: + err = CRYPT_INVALID_ARG; + goto LBL_ERR; + } + } + + /* calc header size */ + z = y; + if (y < 128) { + y += 2; + } else if (y < 256) { + /* 0x30 0x81 LL */ + y += 3; + } else if (y < 65536UL) { + /* 0x30 0x82 LL LL */ + y += 4; + } else if (y < 16777216UL) { + /* 0x30 0x83 LL LL LL */ + y += 5; + } else { + err = CRYPT_INVALID_ARG; + goto LBL_ERR; + } + + /* too big ? */ + if (*outlen < y) { + *outlen = y; + err = CRYPT_BUFFER_OVERFLOW; + goto LBL_ERR; + } + + /* store header */ + x = 0; + out[x++] = (type_of == LTC_ASN1_SEQUENCE) ? 0x30 : 0x31; + + if (z < 128) { + out[x++] = (unsigned char)z; + } else if (z < 256) { + out[x++] = 0x81; + out[x++] = (unsigned char)z; + } else if (z < 65536UL) { + out[x++] = 0x82; + out[x++] = (unsigned char)((z >> 8UL) & 255); + out[x++] = (unsigned char)(z & 255); + } else if (z < 16777216UL) { + out[x++] = 0x83; + out[x++] = (unsigned char)((z >> 16UL) & 255); + out[x++] = (unsigned char)((z >> 8UL) & 255); + out[x++] = (unsigned char)(z & 255); + } + + /* store data */ + *outlen -= x; + for (i = 0; i < inlen; i++) { + type = list[i].type; + size = list[i].size; + data = list[i].data; + + if (type == LTC_ASN1_EOL) { + break; + } + + switch (type) { + case LTC_ASN1_BOOLEAN: + z = *outlen; + if ((err = der_encode_boolean(*((int *)data), out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_INTEGER: + z = *outlen; + if ((err = der_encode_integer(data, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_SHORT_INTEGER: + z = *outlen; + if ((err = der_encode_short_integer(*((unsigned long *)data), out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_BIT_STRING: + z = *outlen; + if ((err = der_encode_bit_string(data, size, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_OCTET_STRING: + z = *outlen; + if ((err = der_encode_octet_string(data, size, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_NULL: + out[x++] = 0x05; + out[x++] = 0x00; + *outlen -= 2; + break; + + case LTC_ASN1_OBJECT_IDENTIFIER: + z = *outlen; + if ((err = der_encode_object_identifier(data, size, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_IA5_STRING: + z = *outlen; + if ((err = der_encode_ia5_string(data, size, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_PRINTABLE_STRING: + z = *outlen; + if ((err = der_encode_printable_string(data, size, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_UTF8_STRING: + z = *outlen; + if ((err = der_encode_utf8_string(data, size, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_UTCTIME: + z = *outlen; + if ((err = der_encode_utctime(data, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_SET: + z = *outlen; + if ((err = der_encode_set(data, size, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_SETOF: + z = *outlen; + if ((err = der_encode_setof(data, size, out + x, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + case LTC_ASN1_SEQUENCE: + z = *outlen; + if ((err = der_encode_sequence_ex(data, size, out + x, &z, type)) != CRYPT_OK) { + goto LBL_ERR; + } + x += z; + *outlen -= z; + break; + + default: + err = CRYPT_INVALID_ARG; + goto LBL_ERR; + } + } + *outlen = x; + err = CRYPT_OK; + +LBL_ERR: + return err; +} +#endif + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +#include + + +/** + @file der_encode_sequence_multi.c + ASN.1 DER, encode a SEQUENCE, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Encode a SEQUENCE type using a VA list + @param out [out] Destination for data + @param outlen [in/out] Length of buffer and resulting length of output + @remark <...> is of the form (int, unsigned long, void*) + @return CRYPT_OK on success + */ +int der_encode_sequence_multi(unsigned char *out, unsigned long *outlen, ...) { + int err, type; + unsigned long size, x; + void *data; + va_list args; + ltc_asn1_list *list; + + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* get size of output that will be required */ + va_start(args, outlen); + x = 0; + for ( ; ; ) { + type = va_arg(args, int); + size = va_arg(args, unsigned long); + data = va_arg(args, void *); + + if (type == LTC_ASN1_EOL) { + break; + } + + switch (type) { + case LTC_ASN1_BOOLEAN: + case LTC_ASN1_INTEGER: + case LTC_ASN1_SHORT_INTEGER: + case LTC_ASN1_BIT_STRING: + case LTC_ASN1_OCTET_STRING: + case LTC_ASN1_NULL: + case LTC_ASN1_OBJECT_IDENTIFIER: + case LTC_ASN1_IA5_STRING: + case LTC_ASN1_PRINTABLE_STRING: + case LTC_ASN1_UTF8_STRING: + case LTC_ASN1_UTCTIME: + case LTC_ASN1_SEQUENCE: + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + ++x; + break; + + default: + va_end(args); + return CRYPT_INVALID_ARG; + } + } + va_end(args); + + /* allocate structure for x elements */ + if (x == 0) { + return CRYPT_NOP; + } + + list = XCALLOC(x, sizeof(*list)); + if (list == NULL) { + return CRYPT_MEM; + } + + /* fill in the structure */ + va_start(args, outlen); + x = 0; + for ( ; ; ) { + type = va_arg(args, int); + size = va_arg(args, unsigned long); + data = va_arg(args, void *); + + if (type == LTC_ASN1_EOL) { + break; + } + + switch (type) { + case LTC_ASN1_BOOLEAN: + case LTC_ASN1_INTEGER: + case LTC_ASN1_SHORT_INTEGER: + case LTC_ASN1_BIT_STRING: + case LTC_ASN1_OCTET_STRING: + case LTC_ASN1_NULL: + case LTC_ASN1_OBJECT_IDENTIFIER: + case LTC_ASN1_IA5_STRING: + case LTC_ASN1_PRINTABLE_STRING: + case LTC_ASN1_UTF8_STRING: + case LTC_ASN1_UTCTIME: + case LTC_ASN1_SEQUENCE: + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + list[x].type = type; + list[x].size = size; + list[x++].data = data; + break; + + default: + va_end(args); + err = CRYPT_INVALID_ARG; + goto LBL_ERR; + } + } + va_end(args); + + err = der_encode_sequence(list, x, out, outlen); +LBL_ERR: + XFREE(list); + return err; +} +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/sequence/der_encode_sequence_multi.c,v $ */ +/* $Revision: 1.12 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_set.c + ASN.1 DER, Encode a SET, Tom St Denis + */ + +#ifdef LTC_DER + +/* LTC define to ASN.1 TAG */ +static int ltc_to_asn1(int v) { + switch (v) { + case LTC_ASN1_BOOLEAN: + return 0x01; + + case LTC_ASN1_INTEGER: + case LTC_ASN1_SHORT_INTEGER: + return 0x02; + + case LTC_ASN1_BIT_STRING: + return 0x03; + + case LTC_ASN1_OCTET_STRING: + return 0x04; + + case LTC_ASN1_NULL: + return 0x05; + + case LTC_ASN1_OBJECT_IDENTIFIER: + return 0x06; + + case LTC_ASN1_UTF8_STRING: + return 0x0C; + + case LTC_ASN1_PRINTABLE_STRING: + return 0x13; + + case LTC_ASN1_IA5_STRING: + return 0x16; + + case LTC_ASN1_UTCTIME: + return 0x17; + + case LTC_ASN1_SEQUENCE: + return 0x30; + + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + return 0x31; + + default: + return -1; + } +} + +static int qsort_helper_set(const void *a, const void *b) { + ltc_asn1_list *A = (ltc_asn1_list *)a, *B = (ltc_asn1_list *)b; + int r; + + r = ltc_to_asn1(A->type) - ltc_to_asn1(B->type); + + /* for QSORT the order is UNDEFINED if they are "equal" which means it is NOT DETERMINISTIC. So we force it to be :-) */ + if (r == 0) { + /* their order in the original list now determines the position */ + return A->used - B->used; + } else { + return r; + } +} + +/* + Encode a SET type + @param list The list of items to encode + @param inlen The number of items in the list + @param out [out] The destination + @param outlen [in/out] The size of the output + @return CRYPT_OK on success + */ +int der_encode_set(ltc_asn1_list *list, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + ltc_asn1_list *copy; + unsigned long x; + int err; + + /* make copy of list */ + copy = XCALLOC(inlen, sizeof(*copy)); + if (copy == NULL) { + return CRYPT_MEM; + } + + /* fill in used member with index so we can fully sort it */ + for (x = 0; x < inlen; x++) { + copy[x] = list[x]; + copy[x].used = x; + } + + /* sort it by the "type" field */ + XQSORT(copy, inlen, sizeof(*copy), &qsort_helper_set); + + /* call der_encode_sequence_ex() */ + err = der_encode_sequence_ex(copy, inlen, out, outlen, LTC_ASN1_SET); + + /* free list */ + XFREE(copy); + + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/set/der_encode_set.c,v $ */ +/* $Revision: 1.12 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_setof.c + ASN.1 DER, Encode SET OF, Tom St Denis + */ + +#ifdef LTC_DER + +struct edge { + unsigned char *start; + unsigned long size; +}; + +static int qsort_helper(const void *a, const void *b) { + struct edge *A = (struct edge *)a, *B = (struct edge *)b; + int r; + unsigned long x; + + /* compare min length */ + r = XMEMCMP(A->start, B->start, MIN(A->size, B->size)); + + if ((r == 0) && (A->size != B->size)) { + if (A->size > B->size) { + for (x = B->size; x < A->size; x++) { + if (A->start[x]) { + return 1; + } + } + } else { + for (x = A->size; x < B->size; x++) { + if (B->start[x]) { + return -1; + } + } + } + } + + return r; +} + +/** + Encode a SETOF stucture + @param list The list of items to encode + @param inlen The number of items in the list + @param out [out] The destination + @param outlen [in/out] The size of the output + @return CRYPT_OK on success + */ +int der_encode_setof(ltc_asn1_list *list, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long x, y, z, hdrlen; + int err; + struct edge *edges; + unsigned char *ptr, *buf; + + /* check that they're all the same type */ + for (x = 1; x < inlen; x++) { + if (list[x].type != list[x - 1].type) { + return CRYPT_INVALID_ARG; + } + } + + /* alloc buffer to store copy of output */ + buf = XCALLOC(1, *outlen); + if (buf == NULL) { + return CRYPT_MEM; + } + + /* encode list */ + if ((err = der_encode_sequence_ex(list, inlen, buf, outlen, LTC_ASN1_SETOF)) != CRYPT_OK) { + XFREE(buf); + return err; + } + + /* allocate edges */ + edges = XCALLOC(inlen, sizeof(*edges)); + if (edges == NULL) { + XFREE(buf); + return CRYPT_MEM; + } + + /* skip header */ + ptr = buf + 1; + + /* now skip length data */ + x = *ptr++; + if (x >= 0x80) { + ptr += (x & 0x7F); + } + + /* get the size of the static header */ + hdrlen = ((unsigned long)ptr) - ((unsigned long)buf); + + + /* scan for edges */ + x = 0; + while (ptr < (buf + *outlen)) { + /* store start */ + edges[x].start = ptr; + + /* skip type */ + z = 1; + + /* parse length */ + y = ptr[z++]; + if (y < 128) { + edges[x].size = y; + } else { + y &= 0x7F; + edges[x].size = 0; + while (y--) { + edges[x].size = (edges[x].size << 8) | ((unsigned long)ptr[z++]); + } + } + + /* skip content */ + edges[x].size += z; + ptr += edges[x].size; + ++x; + } + + /* sort based on contents (using edges) */ + XQSORT(edges, inlen, sizeof(*edges), &qsort_helper); + + /* copy static header */ + XMEMCPY(out, buf, hdrlen); + + /* copy+sort using edges+indecies to output from buffer */ + for (y = hdrlen, x = 0; x < inlen; x++) { + XMEMCPY(out + y, edges[x].start, edges[x].size); + y += edges[x].size; + } + + #ifdef LTC_CLEAN_STACK + zeromem(buf, *outlen); + #endif + + /* free buffers */ + XFREE(edges); + XFREE(buf); + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/set/der_encode_setof.c,v $ */ +/* $Revision: 1.12 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_short_integer.c + ASN.1 DER, encode an integer, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store a short integer in the range (0,2^32-1) + @param num The integer to encode + @param out [out] The destination for the DER encoded integers + @param outlen [in/out] The max size and resulting size of the DER encoded integers + @return CRYPT_OK if successful + */ +int der_encode_short_integer(unsigned long num, unsigned char *out, unsigned long *outlen) { + unsigned long len, x, y, z; + int err; + + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* force to 32 bits */ + num &= 0xFFFFFFFFUL; + + /* find out how big this will be */ + if ((err = der_length_short_integer(num, &len)) != CRYPT_OK) { + return err; + } + + if (*outlen < len) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + /* get len of output */ + z = 0; + y = num; + while (y) { + ++z; + y >>= 8; + } + + /* handle zero */ + if (z == 0) { + z = 1; + } + + /* see if msb is set */ + z += (num & (1UL << ((z << 3) - 1))) ? 1 : 0; + + /* adjust the number so the msB is non-zero */ + for (x = 0; (z <= 4) && (x < (4 - z)); x++) { + num <<= 8; + } + + /* store header */ + x = 0; + out[x++] = 0x02; + out[x++] = (unsigned char)z; + + /* if 31st bit is set output a leading zero and decrement count */ + if (z == 5) { + out[x++] = 0; + --z; + } + + /* store values */ + for (y = 0; y < z; y++) { + out[x++] = (unsigned char)((num >> 24) & 0xFF); + num <<= 8; + } + + /* we good */ + *outlen = x; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/short_integer/der_encode_short_integer.c,v $ */ +/* $Revision: 1.8 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_utctime.c + ASN.1 DER, encode a UTCTIME, Tom St Denis + */ + +#ifdef LTC_DER + +static const char baseten[] = "0123456789"; + + #define STORE_V(y) \ + out[x++] = der_ia5_char_encode(baseten[(y / 10) % 10]); \ + out[x++] = der_ia5_char_encode(baseten[y % 10]); + +/** + Encodes a UTC time structure in DER format + @param utctime The UTC time structure to encode + @param out The destination of the DER encoding of the UTC time structure + @param outlen [in/out] The length of the DER encoding + @return CRYPT_OK if successful + */ +int der_encode_utctime(ltc_utctime *utctime, + unsigned char *out, unsigned long *outlen) { + unsigned long x, tmplen; + int err; + + LTC_ARGCHK(utctime != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + if ((err = der_length_utctime(utctime, &tmplen)) != CRYPT_OK) { + return err; + } + if (tmplen > *outlen) { + *outlen = tmplen; + return CRYPT_BUFFER_OVERFLOW; + } + + /* store header */ + out[0] = 0x17; + + /* store values */ + x = 2; + STORE_V(utctime->YY); + STORE_V(utctime->MM); + STORE_V(utctime->DD); + STORE_V(utctime->hh); + STORE_V(utctime->mm); + STORE_V(utctime->ss); + + if (utctime->off_mm || utctime->off_hh) { + out[x++] = der_ia5_char_encode(utctime->off_dir ? '-' : '+'); + STORE_V(utctime->off_hh); + STORE_V(utctime->off_mm); + } else { + out[x++] = der_ia5_char_encode('Z'); + } + + /* store length */ + out[1] = (unsigned char)(x - 2); + + /* all good let's return */ + *outlen = x; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/utctime/der_encode_utctime.c,v $ */ +/* $Revision: 1.10 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_encode_utf8_string.c + ASN.1 DER, encode a UTF8 STRING, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Store an UTF8 STRING + @param in The array of UTF8 to store (one per wchar_t) + @param inlen The number of UTF8 to store + @param out [out] The destination for the DER encoded UTF8 STRING + @param outlen [in/out] The max size and resulting size of the DER UTF8 STRING + @return CRYPT_OK if successful + */ +int der_encode_utf8_string(const wchar_t *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen) { + unsigned long x, y, len; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* get the size */ + for (x = len = 0; x < inlen; x++) { + if ((in[x] < 0) || (in[x] > 0x1FFFF)) { + return CRYPT_INVALID_ARG; + } + len += der_utf8_charsize(in[x]); + } + + if (len < 128) { + y = 2 + len; + } else if (len < 256) { + y = 3 + len; + } else if (len < 65536UL) { + y = 4 + len; + } else if (len < 16777216UL) { + y = 5 + len; + } else { + return CRYPT_INVALID_ARG; + } + + /* too big? */ + if (y > *outlen) { + *outlen = len; + return CRYPT_BUFFER_OVERFLOW; + } + + /* encode the header+len */ + x = 0; + out[x++] = 0x0C; + if (len < 128) { + out[x++] = (unsigned char)len; + } else if (len < 256) { + out[x++] = 0x81; + out[x++] = (unsigned char)len; + } else if (len < 65536UL) { + out[x++] = 0x82; + out[x++] = (unsigned char)((len >> 8) & 255); + out[x++] = (unsigned char)(len & 255); + } else if (len < 16777216UL) { + out[x++] = 0x83; + out[x++] = (unsigned char)((len >> 16) & 255); + out[x++] = (unsigned char)((len >> 8) & 255); + out[x++] = (unsigned char)(len & 255); + } else { + return CRYPT_INVALID_ARG; + } + + /* store UTF8 */ + for (y = 0; y < inlen; y++) { + switch (der_utf8_charsize(in[y])) { + case 1: + out[x++] = (unsigned char)in[y]; + break; + + case 2: + out[x++] = 0xC0 | ((in[y] >> 6) & 0x1F); + out[x++] = 0x80 | (in[y] & 0x3F); + break; + + case 3: + out[x++] = 0xE0 | ((in[y] >> 12) & 0x0F); + out[x++] = 0x80 | ((in[y] >> 6) & 0x3F); + out[x++] = 0x80 | (in[y] & 0x3F); + break; + + case 4: + out[x++] = 0xF0 | ((in[y] >> 18) & 0x07); + out[x++] = 0x80 | ((in[y] >> 12) & 0x3F); + out[x++] = 0x80 | ((in[y] >> 6) & 0x3F); + out[x++] = 0x80 | (in[y] & 0x3F); + break; + } + } + + /* retun length */ + *outlen = x; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/utf8/der_encode_utf8_string.c,v $ */ +/* $Revision: 1.9 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_bit_string.c + ASN.1 DER, get length of BIT STRING, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Gets length of DER encoding of BIT STRING + @param nbits The number of bits in the string to encode + @param outlen [out] The length of the DER encoding for the given string + @return CRYPT_OK if successful + */ +int der_length_bit_string(unsigned long nbits, unsigned long *outlen) { + unsigned long nbytes; + + LTC_ARGCHK(outlen != NULL); + + /* get the number of the bytes */ + nbytes = (nbits >> 3) + ((nbits & 7) ? 1 : 0) + 1; + + if (nbytes < 128) { + /* 03 LL PP DD DD DD ... */ + *outlen = 2 + nbytes; + } else if (nbytes < 256) { + /* 03 81 LL PP DD DD DD ... */ + *outlen = 3 + nbytes; + } else if (nbytes < 65536) { + /* 03 82 LL LL PP DD DD DD ... */ + *outlen = 4 + nbytes; + } else { + return CRYPT_INVALID_ARG; + } + + return CRYPT_OK; +} +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/bit/der_length_bit_string.c,v $ */ +/* $Revision: 1.3 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_boolean.c + ASN.1 DER, get length of a BOOLEAN, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Gets length of DER encoding of a BOOLEAN + @param outlen [out] The length of the DER encoding + @return CRYPT_OK if successful + */ +int der_length_boolean(unsigned long *outlen) { + LTC_ARGCHK(outlen != NULL); + *outlen = 3; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/boolean/der_length_boolean.c,v $ */ +/* $Revision: 1.3 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_ia5_string.c + ASN.1 DER, get length of IA5 STRING, Tom St Denis + */ + +#ifdef LTC_DER + +static const struct { + int code, value; +} ia5_table[] = { + { '\0', 0 }, + { '\a', 7 }, + { '\b', 8 }, + { '\t', 9 }, + { '\n', 10 }, + { '\f', 12 }, + { '\r', 13 }, + { ' ', 32 }, + { '!', 33 }, + { '"', 34 }, + { '#', 35 }, + { '$', 36 }, + { '%', 37 }, + { '&', 38 }, + { '\'', 39 }, + { '(', 40 }, + { ')', 41 }, + { '*', 42 }, + { '+', 43 }, + { ',', 44 }, + { '-', 45 }, + { '.', 46 }, + { '/', 47 }, + { '0', 48 }, + { '1', 49 }, + { '2', 50 }, + { '3', 51 }, + { '4', 52 }, + { '5', 53 }, + { '6', 54 }, + { '7', 55 }, + { '8', 56 }, + { '9', 57 }, + { ':', 58 }, + { ';', 59 }, + { '<', 60 }, + { '=', 61 }, + { '>', 62 }, + { '?', 63 }, + { '@', 64 }, + { 'A', 65 }, + { 'B', 66 }, + { 'C', 67 }, + { 'D', 68 }, + { 'E', 69 }, + { 'F', 70 }, + { 'G', 71 }, + { 'H', 72 }, + { 'I', 73 }, + { 'J', 74 }, + { 'K', 75 }, + { 'L', 76 }, + { 'M', 77 }, + { 'N', 78 }, + { 'O', 79 }, + { 'P', 80 }, + { 'Q', 81 }, + { 'R', 82 }, + { 'S', 83 }, + { 'T', 84 }, + { 'U', 85 }, + { 'V', 86 }, + { 'W', 87 }, + { 'X', 88 }, + { 'Y', 89 }, + { 'Z', 90 }, + { '[', 91 }, + { '\\', 92 }, + { ']', 93 }, + { '^', 94 }, + { '_', 95 }, + { '`', 96 }, + { 'a', 97 }, + { 'b', 98 }, + { 'c', 99 }, + { 'd', 100 }, + { 'e', 101 }, + { 'f', 102 }, + { 'g', 103 }, + { 'h', 104 }, + { 'i', 105 }, + { 'j', 106 }, + { 'k', 107 }, + { 'l', 108 }, + { 'm', 109 }, + { 'n', 110 }, + { 'o', 111 }, + { 'p', 112 }, + { 'q', 113 }, + { 'r', 114 }, + { 's', 115 }, + { 't', 116 }, + { 'u', 117 }, + { 'v', 118 }, + { 'w', 119 }, + { 'x', 120 }, + { 'y', 121 }, + { 'z', 122 }, + { '{', 123 }, + { '|', 124 }, + { '}', 125 }, + { '~', 126 } +}; + +int der_ia5_char_encode(int c) { + int x; + + for (x = 0; x < (int)(sizeof(ia5_table) / sizeof(ia5_table[0])); x++) { + if (ia5_table[x].code == c) { + return ia5_table[x].value; + } + } + return -1; +} + +int der_ia5_value_decode(int v) { + int x; + + for (x = 0; x < (int)(sizeof(ia5_table) / sizeof(ia5_table[0])); x++) { + if (ia5_table[x].value == v) { + return ia5_table[x].code; + } + } + return -1; +} + +/** + Gets length of DER encoding of IA5 STRING + @param octets The values you want to encode + @param noctets The number of octets in the string to encode + @param outlen [out] The length of the DER encoding for the given string + @return CRYPT_OK if successful + */ +int der_length_ia5_string(const unsigned char *octets, unsigned long noctets, unsigned long *outlen) { + unsigned long x; + + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(octets != NULL); + + /* scan string for validity */ + for (x = 0; x < noctets; x++) { + if (der_ia5_char_encode(octets[x]) == -1) { + return CRYPT_INVALID_ARG; + } + } + + if (noctets < 128) { + /* 16 LL DD DD DD ... */ + *outlen = 2 + noctets; + } else if (noctets < 256) { + /* 16 81 LL DD DD DD ... */ + *outlen = 3 + noctets; + } else if (noctets < 65536UL) { + /* 16 82 LL LL DD DD DD ... */ + *outlen = 4 + noctets; + } else if (noctets < 16777216UL) { + /* 16 83 LL LL LL DD DD DD ... */ + *outlen = 5 + noctets; + } else { + return CRYPT_INVALID_ARG; + } + + return CRYPT_OK; +} +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/ia5/der_length_ia5_string.c,v $ */ +/* $Revision: 1.3 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_integer.c + ASN.1 DER, get length of encoding, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Gets length of DER encoding of num + @param num The int to get the size of + @param outlen [out] The length of the DER encoding for the given integer + @return CRYPT_OK if successful + */ +int der_length_integer(void *num, unsigned long *outlen) { + unsigned long z, len; + int leading_zero; + + LTC_ARGCHK(num != NULL); + LTC_ARGCHK(outlen != NULL); + + if (mp_cmp_d(num, 0) != LTC_MP_LT) { + /* positive */ + + /* we only need a leading zero if the msb of the first byte is one */ + if (((mp_count_bits(num) & 7) == 0) || (mp_iszero(num) == LTC_MP_YES)) { + leading_zero = 1; + } else { + leading_zero = 0; + } + + /* size for bignum */ + z = len = leading_zero + mp_unsigned_bin_size(num); + } else { + /* it's negative */ + /* find power of 2 that is a multiple of eight and greater than count bits */ + leading_zero = 0; + z = mp_count_bits(num); + z = z + (8 - (z & 7)); + if (((mp_cnt_lsb(num) + 1) == mp_count_bits(num)) && ((mp_count_bits(num) & 7) == 0)) --z; + len = z = z >> 3; + } + + /* now we need a length */ + if (z < 128) { + /* short form */ + ++len; + } else { + /* long form (relies on z != 0), assumes length bytes < 128 */ + ++len; + + while (z) { + ++len; + z >>= 8; + } + } + + /* we need a 0x02 to indicate it's INTEGER */ + ++len; + + /* return length */ + *outlen = len; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/integer/der_length_integer.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_object_identifier.c + ASN.1 DER, get length of Object Identifier, Tom St Denis + */ + +#ifdef LTC_DER + +unsigned long der_object_identifier_bits(unsigned long x) { + unsigned long c; + + x &= 0xFFFFFFFF; + c = 0; + while (x) { + ++c; + x >>= 1; + } + return c; +} + +/** + Gets length of DER encoding of Object Identifier + @param nwords The number of OID words + @param words The actual OID words to get the size of + @param outlen [out] The length of the DER encoding for the given string + @return CRYPT_OK if successful + */ +int der_length_object_identifier(unsigned long *words, unsigned long nwords, unsigned long *outlen) { + unsigned long y, z, t, wordbuf; + + LTC_ARGCHK(words != NULL); + LTC_ARGCHK(outlen != NULL); + + + /* must be >= 2 words */ + if (nwords < 2) { + return CRYPT_INVALID_ARG; + } + + /* word1 = 0,1,2,3 and word2 0..39 */ + if ((words[0] > 3) || ((words[0] < 2) && (words[1] > 39))) { + return CRYPT_INVALID_ARG; + } + + /* leading word is the first two */ + z = 0; + wordbuf = words[0] * 40 + words[1]; + for (y = 1; y < nwords; y++) { + t = der_object_identifier_bits(wordbuf); + z += t / 7 + ((t % 7) ? 1 : 0) + (wordbuf == 0 ? 1 : 0); + if (y < nwords - 1) { + /* grab next word */ + wordbuf = words[y + 1]; + } + } + + /* now depending on the length our length encoding changes */ + if (z < 128) { + z += 2; + } else if (z < 256) { + z += 3; + } else if (z < 65536UL) { + z += 4; + } else { + return CRYPT_INVALID_ARG; + } + + *outlen = z; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/object_identifier/der_length_object_identifier.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_octet_string.c + ASN.1 DER, get length of OCTET STRING, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Gets length of DER encoding of OCTET STRING + @param noctets The number of octets in the string to encode + @param outlen [out] The length of the DER encoding for the given string + @return CRYPT_OK if successful + */ +int der_length_octet_string(unsigned long noctets, unsigned long *outlen) { + LTC_ARGCHK(outlen != NULL); + + if (noctets < 128) { + /* 04 LL DD DD DD ... */ + *outlen = 2 + noctets; + } else if (noctets < 256) { + /* 04 81 LL DD DD DD ... */ + *outlen = 3 + noctets; + } else if (noctets < 65536UL) { + /* 04 82 LL LL DD DD DD ... */ + *outlen = 4 + noctets; + } else if (noctets < 16777216UL) { + /* 04 83 LL LL LL DD DD DD ... */ + *outlen = 5 + noctets; + } else { + return CRYPT_INVALID_ARG; + } + + return CRYPT_OK; +} +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/octet/der_length_octet_string.c,v $ */ +/* $Revision: 1.3 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_printable_string.c + ASN.1 DER, get length of Printable STRING, Tom St Denis + */ + +#ifdef LTC_DER + +static const struct { + int code, value; +} printable_table[] = { + { ' ', 32 }, + { '\'', 39 }, + { '(', 40 }, + { ')', 41 }, + { '+', 43 }, + { ',', 44 }, + { '-', 45 }, + { '.', 46 }, + { '/', 47 }, + { '0', 48 }, + { '1', 49 }, + { '2', 50 }, + { '3', 51 }, + { '4', 52 }, + { '5', 53 }, + { '6', 54 }, + { '7', 55 }, + { '8', 56 }, + { '9', 57 }, + { ':', 58 }, + { '=', 61 }, + { '?', 63 }, + { 'A', 65 }, + { 'B', 66 }, + { 'C', 67 }, + { 'D', 68 }, + { 'E', 69 }, + { 'F', 70 }, + { 'G', 71 }, + { 'H', 72 }, + { 'I', 73 }, + { 'J', 74 }, + { 'K', 75 }, + { 'L', 76 }, + { 'M', 77 }, + { 'N', 78 }, + { 'O', 79 }, + { 'P', 80 }, + { 'Q', 81 }, + { 'R', 82 }, + { 'S', 83 }, + { 'T', 84 }, + { 'U', 85 }, + { 'V', 86 }, + { 'W', 87 }, + { 'X', 88 }, + { 'Y', 89 }, + { 'Z', 90 }, + { 'a', 97 }, + { 'b', 98 }, + { 'c', 99 }, + { 'd', 100 }, + { 'e', 101 }, + { 'f', 102 }, + { 'g', 103 }, + { 'h', 104 }, + { 'i', 105 }, + { 'j', 106 }, + { 'k', 107 }, + { 'l', 108 }, + { 'm', 109 }, + { 'n', 110 }, + { 'o', 111 }, + { 'p', 112 }, + { 'q', 113 }, + { 'r', 114 }, + { 's', 115 }, + { 't', 116 }, + { 'u', 117 }, + { 'v', 118 }, + { 'w', 119 }, + { 'x', 120 }, + { 'y', 121 }, + { 'z', 122 }, +}; + +int der_printable_char_encode(int c) { + int x; + + for (x = 0; x < (int)(sizeof(printable_table) / sizeof(printable_table[0])); x++) { + if (printable_table[x].code == c) { + return printable_table[x].value; + } + } + return -1; +} + +int der_printable_value_decode(int v) { + int x; + + for (x = 0; x < (int)(sizeof(printable_table) / sizeof(printable_table[0])); x++) { + if (printable_table[x].value == v) { + return printable_table[x].code; + } + } + return -1; +} + +/** + Gets length of DER encoding of Printable STRING + @param octets The values you want to encode + @param noctets The number of octets in the string to encode + @param outlen [out] The length of the DER encoding for the given string + @return CRYPT_OK if successful + */ +int der_length_printable_string(const unsigned char *octets, unsigned long noctets, unsigned long *outlen) { + unsigned long x; + + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(octets != NULL); + + /* scan string for validity */ + for (x = 0; x < noctets; x++) { + if (der_printable_char_encode(octets[x]) == -1) { + return CRYPT_INVALID_ARG; + } + } + + if (noctets < 128) { + /* 16 LL DD DD DD ... */ + *outlen = 2 + noctets; + } else if (noctets < 256) { + /* 16 81 LL DD DD DD ... */ + *outlen = 3 + noctets; + } else if (noctets < 65536UL) { + /* 16 82 LL LL DD DD DD ... */ + *outlen = 4 + noctets; + } else if (noctets < 16777216UL) { + /* 16 83 LL LL LL DD DD DD ... */ + *outlen = 5 + noctets; + } else { + return CRYPT_INVALID_ARG; + } + + return CRYPT_OK; +} +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/printable_string/der_length_printable_string.c,v $ */ +/* $Revision: 1.3 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_sequence.c + ASN.1 DER, length a SEQUENCE, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Get the length of a DER sequence + @param list The sequences of items in the SEQUENCE + @param inlen The number of items + @param outlen [out] The length required in octets to store it + @return CRYPT_OK on success + */ +int der_length_sequence(ltc_asn1_list *list, unsigned long inlen, + unsigned long *outlen) { + int err, type; + unsigned long size, x, y, z, i; + void *data; + + LTC_ARGCHK(list != NULL); + LTC_ARGCHK(outlen != NULL); + + /* get size of output that will be required */ + y = 0; + for (i = 0; i < inlen; i++) { + type = list[i].type; + size = list[i].size; + data = list[i].data; + + if (type == LTC_ASN1_EOL) { + break; + } + + switch (type) { + case LTC_ASN1_BOOLEAN: + if ((err = der_length_boolean(&x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_INTEGER: + if ((err = der_length_integer(data, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_SHORT_INTEGER: + if ((err = der_length_short_integer(*((unsigned long *)data), &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_BIT_STRING: + if ((err = der_length_bit_string(size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_OCTET_STRING: + if ((err = der_length_octet_string(size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_NULL: + y += 2; + break; + + case LTC_ASN1_OBJECT_IDENTIFIER: + if ((err = der_length_object_identifier(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_IA5_STRING: + if ((err = der_length_ia5_string(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_PRINTABLE_STRING: + if ((err = der_length_printable_string(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_UTCTIME: + if ((err = der_length_utctime(data, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_UTF8_STRING: + if ((err = der_length_utf8_string(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + case LTC_ASN1_SEQUENCE: + if ((err = der_length_sequence(data, size, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + y += x; + break; + + + default: + err = CRYPT_INVALID_ARG; + goto LBL_ERR; + } + } + + /* calc header size */ + z = y; + if (y < 128) { + y += 2; + } else if (y < 256) { + /* 0x30 0x81 LL */ + y += 3; + } else if (y < 65536UL) { + /* 0x30 0x82 LL LL */ + y += 4; + } else if (y < 16777216UL) { + /* 0x30 0x83 LL LL LL */ + y += 5; + } else { + err = CRYPT_INVALID_ARG; + goto LBL_ERR; + } + + /* store size */ + *outlen = y; + err = CRYPT_OK; + +LBL_ERR: + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/sequence/der_length_sequence.c,v $ */ +/* $Revision: 1.14 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_short_integer.c + ASN.1 DER, get length of encoding, Tom St Denis + */ + + +#ifdef LTC_DER + +/** + Gets length of DER encoding of num + @param num The integer to get the size of + @param outlen [out] The length of the DER encoding for the given integer + @return CRYPT_OK if successful + */ +int der_length_short_integer(unsigned long num, unsigned long *outlen) { + unsigned long z, y, len; + + LTC_ARGCHK(outlen != NULL); + + /* force to 32 bits */ + num &= 0xFFFFFFFFUL; + + /* get the number of bytes */ + z = 0; + y = num; + while (y) { + ++z; + y >>= 8; + } + + /* handle zero */ + if (z == 0) { + z = 1; + } + + /* we need a 0x02 to indicate it's INTEGER */ + len = 1; + + /* length byte */ + ++len; + + /* bytes in value */ + len += z; + + /* see if msb is set */ + len += (num & (1UL << ((z << 3) - 1))) ? 1 : 0; + + /* return length */ + *outlen = len; + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/short_integer/der_length_short_integer.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_utctime.c + ASN.1 DER, get length of UTCTIME, Tom St Denis + */ + +#ifdef LTC_DER + +/** + Gets length of DER encoding of UTCTIME + @param utctime The UTC time structure to get the size of + @param outlen [out] The length of the DER encoding + @return CRYPT_OK if successful + */ +int der_length_utctime(ltc_utctime *utctime, unsigned long *outlen) { + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(utctime != NULL); + + if ((utctime->off_hh == 0) && (utctime->off_mm == 0)) { + /* we encode as YYMMDDhhmmssZ */ + *outlen = 2 + 13; + } else { + /* we encode as YYMMDDhhmmss{+|-}hh'mm' */ + *outlen = 2 + 17; + } + + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/utctime/der_length_utctime.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_length_utf8_string.c + ASN.1 DER, get length of UTF8 STRING, Tom St Denis + */ + +#ifdef LTC_DER + +/** Return the size in bytes of a UTF-8 character + @param c The UTF-8 character to measure + @return The size in bytes + */ +unsigned long der_utf8_charsize(const wchar_t c) { + if (c <= 0x7F) { + return 1; + } else if (c <= 0x7FF) { + return 2; + } else if (c <= 0xFFFF) { + return 3; + } else { + return 4; + } +} + +/** + Gets length of DER encoding of UTF8 STRING + @param in The characters to measure the length of + @param noctets The number of octets in the string to encode + @param outlen [out] The length of the DER encoding for the given string + @return CRYPT_OK if successful + */ +int der_length_utf8_string(const wchar_t *in, unsigned long noctets, unsigned long *outlen) { + unsigned long x, len; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(outlen != NULL); + + len = 0; + for (x = 0; x < noctets; x++) { + if ((in[x] < 0) || (in[x] > 0x10FFFF)) { + return CRYPT_INVALID_ARG; + } + len += der_utf8_charsize(in[x]); + } + + if (len < 128) { + /* 0C LL DD DD DD ... */ + *outlen = 2 + len; + } else if (len < 256) { + /* 0C 81 LL DD DD DD ... */ + *outlen = 3 + len; + } else if (len < 65536UL) { + /* 0C 82 LL LL DD DD DD ... */ + *outlen = 4 + len; + } else if (len < 16777216UL) { + /* 0C 83 LL LL LL DD DD DD ... */ + *outlen = 5 + len; + } else { + return CRYPT_INVALID_ARG; + } + + return CRYPT_OK; +} +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/utf8/der_length_utf8_string.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file der_sequence_free.c + ASN.1 DER, free's a structure allocated by der_decode_sequence_flexi(), Tom St Denis + */ + +#ifdef LTC_DER + +/** + Free memory allocated by der_decode_sequence_flexi() + @param in The list to free + */ +void der_sequence_free(ltc_asn1_list *in) { + ltc_asn1_list *l; + + /* walk to the start of the chain */ + while (in->prev != NULL || in->parent != NULL) { + if (in->parent != NULL) { + in = in->parent; + } else { + in = in->prev; + } + } + + /* now walk the list and free stuff */ + while (in != NULL) { + /* is there a child? */ + if (in->child) { + /* disconnect */ + in->child->parent = NULL; + der_sequence_free(in->child); + } + + switch (in->type) { + case LTC_ASN1_SET: + case LTC_ASN1_SETOF: + case LTC_ASN1_SEQUENCE: + break; + + case LTC_ASN1_INTEGER: + if (in->data != NULL) { + mp_clear(in->data); + } + break; + + default: + if (in->data != NULL) { + XFREE(in->data); + } + } + + /* move to next and free current */ + l = in->next; + XFREE(in); + in = l; + } +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/asn1/der/sequence/der_sequence_free.c,v $ */ +/* $Revision: 1.4 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/* This holds the key settings. ***MUST*** be organized by size from smallest to largest. */ +const ltc_ecc_set_type ltc_ecc_sets[] = { + #ifdef ECC112 + { + 14, + "SECP112R1", + "DB7C2ABF62E35E668076BEAD208B", + "659EF8BA043916EEDE8911702B22", + "DB7C2ABF62E35E7628DFAC6561C5", + "09487239995A5EE76B55F9C2F098", + "A89CE5AF8724C0A23E0E0FF77500" + }, + #endif + #ifdef ECC128 + { + 16, + "SECP128R1", + "FFFFFFFDFFFFFFFFFFFFFFFFFFFFFFFF", + "E87579C11079F43DD824993C2CEE5ED3", + "FFFFFFFE0000000075A30D1B9038A115", + "161FF7528B899B2D0C28607CA52C5B86", + "CF5AC8395BAFEB13C02DA292DDED7A83", + }, + #endif + #ifdef ECC160 + { + 20, + "SECP160R1", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7FFFFFFF", + "1C97BEFC54BD7A8B65ACF89F81D4D4ADC565FA45", + "0100000000000000000001F4C8F927AED3CA752257", + "4A96B5688EF573284664698968C38BB913CBFC82", + "23A628553168947D59DCC912042351377AC5FB32", + }, + #endif + #ifdef ECC192 + { + 24, + "ECC-192", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFF", + "64210519E59C80E70FA7E9AB72243049FEB8DEECC146B9B1", + "FFFFFFFFFFFFFFFFFFFFFFFF99DEF836146BC9B1B4D22831", + "188DA80EB03090F67CBF20EB43A18800F4FF0AFD82FF1012", + "7192B95FFC8DA78631011ED6B24CDD573F977A11E794811", + }, + #endif + #ifdef ECC224 + { + 28, + "ECC-224", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF000000000000000000000001", + "B4050A850C04B3ABF54132565044B0B7D7BFD8BA270B39432355FFB4", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFF16A2E0B8F03E13DD29455C5C2A3D", + "B70E0CBD6BB4BF7F321390B94A03C1D356C21122343280D6115C1D21", + "BD376388B5F723FB4C22DFE6CD4375A05A07476444D5819985007E34", + }, + #endif + #ifdef ECC256 + { + 32, + "ECC-256", + "FFFFFFFF00000001000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFF", + "5AC635D8AA3A93E7B3EBBD55769886BC651D06B0CC53B0F63BCE3C3E27D2604B", + "FFFFFFFF00000000FFFFFFFFFFFFFFFFBCE6FAADA7179E84F3B9CAC2FC632551", + "6B17D1F2E12C4247F8BCE6E563A440F277037D812DEB33A0F4A13945D898C296", + "4FE342E2FE1A7F9B8EE7EB4A7C0F9E162BCE33576B315ECECBB6406837BF51F5", + }, + #endif + #ifdef ECC384 + { + 48, + "ECC-384", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFF0000000000000000FFFFFFFF", + "B3312FA7E23EE7E4988E056BE3F82D19181D9C6EFE8141120314088F5013875AC656398D8A2ED19D2A85C8EDD3EC2AEF", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFC7634D81F4372DDF581A0DB248B0A77AECEC196ACCC52973", + "AA87CA22BE8B05378EB1C71EF320AD746E1D3B628BA79B9859F741E082542A385502F25DBF55296C3A545E3872760AB7", + "3617DE4A96262C6F5D9E98BF9292DC29F8F41DBD289A147CE9DA3113B5F0B8C00A60B1CE1D7E819D7A431D7C90EA0E5F", + }, + #endif + #ifdef ECC521 + { + 66, + "ECC-521", + "1FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF", + "51953EB9618E1C9A1F929A21A0B68540EEA2DA725B99B315F3B8B489918EF109E156193951EC7E937B1652C0BD3BB1BF073573DF883D2C34F1EF451FD46B503F00", + "1FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFA51868783BF2F966B7FCC0148F709A5D03BB5C9B8899C47AEBB6FB71E91386409", + "C6858E06B70404E9CD9E3ECB662395B4429C648139053FB521F828AF606B4D3DBAA14B5E77EFE75928FE1DC127A2FFA8DE3348B3C1856A429BF97E7E31C2E5BD66", + "11839296A789A3BC0045C8A5FB42C7D1BD998F54449579B446817AFBD17273E662C97EE72995EF42640C550B9013FAD0761353C7086A272C24088BE94769FD16650", + }, + #endif + { + 0, + NULL, NULL, NULL, NULL, NULL, NULL + } +}; +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc.c,v $ */ +/* $Revision: 1.40 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_ansi_x963_export.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** ECC X9.63 (Sec. 4.3.6) uncompressed export + @param key Key to export + @param out [out] destination of export + @param outlen [in/out] Length of destination and final output size + Return CRYPT_OK on success + */ +int ecc_ansi_x963_export(ecc_key *key, unsigned char *out, unsigned long *outlen) { + unsigned char buf[ECC_BUF_SIZE]; + unsigned long numlen; + + LTC_ARGCHK(key != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + if (ltc_ecc_is_valid_idx(key->idx) == 0) { + return CRYPT_INVALID_ARG; + } + numlen = key->dp->size; + + if (*outlen < (1 + 2 * numlen)) { + *outlen = 1 + 2 * numlen; + return CRYPT_BUFFER_OVERFLOW; + } + + /* store byte 0x04 */ + out[0] = 0x04; + + /* pad and store x */ + zeromem(buf, sizeof(buf)); + mp_to_unsigned_bin(key->pubkey.x, buf + (numlen - mp_unsigned_bin_size(key->pubkey.x))); + XMEMCPY(out + 1, buf, numlen); + + /* pad and store y */ + zeromem(buf, sizeof(buf)); + mp_to_unsigned_bin(key->pubkey.y, buf + (numlen - mp_unsigned_bin_size(key->pubkey.y))); + XMEMCPY(out + 1 + numlen, buf, numlen); + + *outlen = 1 + 2 * numlen; + return CRYPT_OK; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_ansi_x963_export.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_ansi_x963_import.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** Import an ANSI X9.63 format public key + @param in The input data to read + @param inlen The length of the input data + @param key [out] destination to store imported key \ + */ +int ecc_ansi_x963_import(const unsigned char *in, unsigned long inlen, ecc_key *key) { + return ecc_ansi_x963_import_ex(in, inlen, key, NULL); +} + +int ecc_ansi_x963_import_ex(const unsigned char *in, unsigned long inlen, ecc_key *key, ltc_ecc_set_type *dp) { + int x, err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(key != NULL); + + /* must be odd */ + if ((inlen & 1) == 0) { + return CRYPT_INVALID_ARG; + } + + /* init key */ + if (mp_init_multi(&key->pubkey.x, &key->pubkey.y, &key->pubkey.z, &key->k, NULL) != CRYPT_OK) { + return CRYPT_MEM; + } + + /* check for 4, 6 or 7 */ + if ((in[0] != 4) && (in[0] != 6) && (in[0] != 7)) { + err = CRYPT_INVALID_PACKET; + goto error; + } + + /* read data */ + if ((err = mp_read_unsigned_bin(key->pubkey.x, (unsigned char *)in + 1, (inlen - 1) >> 1)) != CRYPT_OK) { + goto error; + } + + if ((err = mp_read_unsigned_bin(key->pubkey.y, (unsigned char *)in + 1 + ((inlen - 1) >> 1), (inlen - 1) >> 1)) != CRYPT_OK) { + goto error; + } + if ((err = mp_set(key->pubkey.z, 1)) != CRYPT_OK) { + goto error; + } + + if (dp == NULL) { + /* determine the idx */ + for (x = 0; ltc_ecc_sets[x].size != 0; x++) { + if ((unsigned)ltc_ecc_sets[x].size >= ((inlen - 1) >> 1)) { + break; + } + } + if (ltc_ecc_sets[x].size == 0) { + err = CRYPT_INVALID_PACKET; + goto error; + } + /* set the idx */ + key->idx = x; + key->dp = <c_ecc_sets[x]; + } else { + if (((inlen - 1) >> 1) != (unsigned long)dp->size) { + err = CRYPT_INVALID_PACKET; + goto error; + } + key->idx = -1; + key->dp = dp; + } + key->type = PK_PUBLIC; + + /* we're done */ + return CRYPT_OK; +error: + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_ansi_x963_import.c,v $ */ +/* $Revision: 1.11 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_decrypt_key.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Decrypt an ECC encrypted key + @param in The ciphertext + @param inlen The length of the ciphertext (octets) + @param out [out] The plaintext + @param outlen [in/out] The max size and resulting size of the plaintext + @param key The corresponding private ECC key + @return CRYPT_OK if successful + */ +int ecc_decrypt_key(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + ecc_key *key) { + unsigned char *ecc_shared, *skey, *pub_expt; + unsigned long x, y, hashOID[32]; + int hash, err; + ecc_key pubkey; + ltc_asn1_list decode[3]; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(key != NULL); + + /* right key type? */ + if (key->type != PK_PRIVATE) { + return CRYPT_PK_NOT_PRIVATE; + } + + /* decode to find out hash */ + LTC_SET_ASN1(decode, 0, LTC_ASN1_OBJECT_IDENTIFIER, hashOID, sizeof(hashOID) / sizeof(hashOID[0])); + + if ((err = der_decode_sequence(in, inlen, decode, 1)) != CRYPT_OK) { + return err; + } + + hash = find_hash_oid(hashOID, decode[0].size); + if (hash_is_valid(hash) != CRYPT_OK) { + return CRYPT_INVALID_PACKET; + } + + /* we now have the hash! */ + + /* allocate memory */ + pub_expt = XMALLOC(ECC_BUF_SIZE); + ecc_shared = XMALLOC(ECC_BUF_SIZE); + skey = XMALLOC(MAXBLOCKSIZE); + if ((pub_expt == NULL) || (ecc_shared == NULL) || (skey == NULL)) { + if (pub_expt != NULL) { + XFREE(pub_expt); + } + if (ecc_shared != NULL) { + XFREE(ecc_shared); + } + if (skey != NULL) { + XFREE(skey); + } + return CRYPT_MEM; + } + LTC_SET_ASN1(decode, 1, LTC_ASN1_OCTET_STRING, pub_expt, ECC_BUF_SIZE); + LTC_SET_ASN1(decode, 2, LTC_ASN1_OCTET_STRING, skey, MAXBLOCKSIZE); + + /* read the structure in now */ + if ((err = der_decode_sequence(in, inlen, decode, 3)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* import ECC key from packet */ + if ((err = ecc_import(decode[1].data, decode[1].size, &pubkey)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* make shared key */ + x = ECC_BUF_SIZE; + if ((err = ecc_shared_secret(key, &pubkey, ecc_shared, &x)) != CRYPT_OK) { + ecc_free(&pubkey); + goto LBL_ERR; + } + ecc_free(&pubkey); + + y = MIN(ECC_BUF_SIZE, MAXBLOCKSIZE); + if ((err = hash_memory(hash, ecc_shared, x, ecc_shared, &y)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* ensure the hash of the shared secret is at least as big as the encrypt itself */ + if (decode[2].size > y) { + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + + /* avoid buffer overflow */ + if (*outlen < decode[2].size) { + *outlen = decode[2].size; + err = CRYPT_BUFFER_OVERFLOW; + goto LBL_ERR; + } + + /* Decrypt the key */ + for (x = 0; x < decode[2].size; x++) { + out[x] = skey[x] ^ ecc_shared[x]; + } + *outlen = x; + + err = CRYPT_OK; +LBL_ERR: + #ifdef LTC_CLEAN_STACK + zeromem(pub_expt, ECC_BUF_SIZE); + zeromem(ecc_shared, ECC_BUF_SIZE); + zeromem(skey, MAXBLOCKSIZE); + #endif + + XFREE(pub_expt); + XFREE(ecc_shared); + XFREE(skey); + + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_decrypt_key.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_encrypt_key.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Encrypt a symmetric key with ECC + @param in The symmetric key you want to encrypt + @param inlen The length of the key to encrypt (octets) + @param out [out] The destination for the ciphertext + @param outlen [in/out] The max size and resulting size of the ciphertext + @param prng An active PRNG state + @param wprng The index of the PRNG you wish to use + @param hash The index of the hash you want to use + @param key The ECC key you want to encrypt to + @return CRYPT_OK if successful + */ +int ecc_encrypt_key(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + prng_state *prng, int wprng, int hash, + ecc_key *key) { + unsigned char *pub_expt, *ecc_shared, *skey; + ecc_key pubkey; + unsigned long x, y, pubkeysize; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(key != NULL); + + /* check that wprng/cipher/hash are not invalid */ + if ((err = prng_is_valid(wprng)) != CRYPT_OK) { + return err; + } + + if ((err = hash_is_valid(hash)) != CRYPT_OK) { + return err; + } + + if (inlen > hash_descriptor[hash].hashsize) { + return CRYPT_INVALID_HASH; + } + + /* make a random key and export the public copy */ + if ((err = ecc_make_key_ex(prng, wprng, &pubkey, key->dp)) != CRYPT_OK) { + return err; + } + + pub_expt = XMALLOC(ECC_BUF_SIZE); + ecc_shared = XMALLOC(ECC_BUF_SIZE); + skey = XMALLOC(MAXBLOCKSIZE); + if ((pub_expt == NULL) || (ecc_shared == NULL) || (skey == NULL)) { + if (pub_expt != NULL) { + XFREE(pub_expt); + } + if (ecc_shared != NULL) { + XFREE(ecc_shared); + } + if (skey != NULL) { + XFREE(skey); + } + ecc_free(&pubkey); + return CRYPT_MEM; + } + + pubkeysize = ECC_BUF_SIZE; + if ((err = ecc_export(pub_expt, &pubkeysize, PK_PUBLIC, &pubkey)) != CRYPT_OK) { + ecc_free(&pubkey); + goto LBL_ERR; + } + + /* make random key */ + x = ECC_BUF_SIZE; + if ((err = ecc_shared_secret(&pubkey, key, ecc_shared, &x)) != CRYPT_OK) { + ecc_free(&pubkey); + goto LBL_ERR; + } + ecc_free(&pubkey); + y = MAXBLOCKSIZE; + if ((err = hash_memory(hash, ecc_shared, x, skey, &y)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* Encrypt key */ + for (x = 0; x < inlen; x++) { + skey[x] ^= in[x]; + } + + err = der_encode_sequence_multi(out, outlen, + LTC_ASN1_OBJECT_IDENTIFIER, hash_descriptor[hash].OIDlen, hash_descriptor[hash].OID, + LTC_ASN1_OCTET_STRING, pubkeysize, pub_expt, + LTC_ASN1_OCTET_STRING, inlen, skey, + LTC_ASN1_EOL, 0UL, NULL); + +LBL_ERR: + #ifdef LTC_CLEAN_STACK + /* clean up */ + zeromem(pub_expt, ECC_BUF_SIZE); + zeromem(ecc_shared, ECC_BUF_SIZE); + zeromem(skey, MAXBLOCKSIZE); + #endif + + XFREE(skey); + XFREE(ecc_shared); + XFREE(pub_expt); + + return err; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_encrypt_key.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_export.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Export an ECC key as a binary packet + @param out [out] Destination for the key + @param outlen [in/out] Max size and resulting size of the exported key + @param type The type of key you want to export (PK_PRIVATE or PK_PUBLIC) + @param key The key to export + @return CRYPT_OK if successful + */ +int ecc_export(unsigned char *out, unsigned long *outlen, int type, ecc_key *key) { + int err; + unsigned char flags[1]; + unsigned long key_size; + + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(key != NULL); + + /* type valid? */ + if ((key->type != PK_PRIVATE) && (type == PK_PRIVATE)) { + return CRYPT_PK_TYPE_MISMATCH; + } + + if (ltc_ecc_is_valid_idx(key->idx) == 0) { + return CRYPT_INVALID_ARG; + } + + /* we store the NIST byte size */ + key_size = key->dp->size; + + if (type == PK_PRIVATE) { + flags[0] = 1; + err = der_encode_sequence_multi(out, outlen, + LTC_ASN1_BIT_STRING, 1UL, flags, + LTC_ASN1_SHORT_INTEGER, 1UL, &key_size, + LTC_ASN1_INTEGER, 1UL, key->pubkey.x, + LTC_ASN1_INTEGER, 1UL, key->pubkey.y, + LTC_ASN1_INTEGER, 1UL, key->k, + LTC_ASN1_EOL, 0UL, NULL); + } else { + flags[0] = 0; + err = der_encode_sequence_multi(out, outlen, + LTC_ASN1_BIT_STRING, 1UL, flags, + LTC_ASN1_SHORT_INTEGER, 1UL, &key_size, + LTC_ASN1_INTEGER, 1UL, key->pubkey.x, + LTC_ASN1_INTEGER, 1UL, key->pubkey.y, + LTC_ASN1_EOL, 0UL, NULL); + } + + return err; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_export.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_free.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Free an ECC key from memory + @param key The key you wish to free + */ +void ecc_free(ecc_key *key) { + LTC_ARGCHKVD(key != NULL); + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_free.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_get_size.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Get the size of an ECC key + @param key The key to get the size of + @return The size (octets) of the key or INT_MAX on error + */ +int ecc_get_size(ecc_key *key) { + LTC_ARGCHK(key != NULL); + if (ltc_ecc_is_valid_idx(key->idx)) + return key->dp->size; + else + return INT_MAX; /* large value known to cause it to fail when passed to ecc_make_key() */ +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_get_size.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_import.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +static int is_point(ecc_key *key) { + void *prime, *b, *t1, *t2; + int err; + + if ((err = mp_init_multi(&prime, &b, &t1, &t2, NULL)) != CRYPT_OK) { + return err; + } + + /* load prime and b */ + if ((err = mp_read_radix(prime, key->dp->prime, 16)) != CRYPT_OK) { + goto error; + } + if ((err = mp_read_radix(b, key->dp->B, 16)) != CRYPT_OK) { + goto error; + } + + /* compute y^2 */ + if ((err = mp_sqr(key->pubkey.y, t1)) != CRYPT_OK) { + goto error; + } + + /* compute x^3 */ + if ((err = mp_sqr(key->pubkey.x, t2)) != CRYPT_OK) { + goto error; + } + if ((err = mp_mod(t2, prime, t2)) != CRYPT_OK) { + goto error; + } + if ((err = mp_mul(key->pubkey.x, t2, t2)) != CRYPT_OK) { + goto error; + } + + /* compute y^2 - x^3 */ + if ((err = mp_sub(t1, t2, t1)) != CRYPT_OK) { + goto error; + } + + /* compute y^2 - x^3 + 3x */ + if ((err = mp_add(t1, key->pubkey.x, t1)) != CRYPT_OK) { + goto error; + } + if ((err = mp_add(t1, key->pubkey.x, t1)) != CRYPT_OK) { + goto error; + } + if ((err = mp_add(t1, key->pubkey.x, t1)) != CRYPT_OK) { + goto error; + } + if ((err = mp_mod(t1, prime, t1)) != CRYPT_OK) { + goto error; + } + while (mp_cmp_d(t1, 0) == LTC_MP_LT) { + if ((err = mp_add(t1, prime, t1)) != CRYPT_OK) { + goto error; + } + } + while (mp_cmp(t1, prime) != LTC_MP_LT) { + if ((err = mp_sub(t1, prime, t1)) != CRYPT_OK) { + goto error; + } + } + + /* compare to b */ + if (mp_cmp(t1, b) != LTC_MP_EQ) { + err = CRYPT_INVALID_PACKET; + } else { + err = CRYPT_OK; + } + +error: + mp_clear_multi(prime, b, t1, t2, NULL); + return err; +} + +/** + Import an ECC key from a binary packet + @param in The packet to import + @param inlen The length of the packet + @param key [out] The destination of the import + @return CRYPT_OK if successful, upon error all allocated memory will be freed + */ +int ecc_import(const unsigned char *in, unsigned long inlen, ecc_key *key) { + return ecc_import_ex(in, inlen, key, NULL); +} + +/** + Import an ECC key from a binary packet, using user supplied domain params rather than one of the NIST ones + @param in The packet to import + @param inlen The length of the packet + @param key [out] The destination of the import + @param dp pointer to user supplied params; must be the same as the params used when exporting + @return CRYPT_OK if successful, upon error all allocated memory will be freed + */ +int ecc_import_ex(const unsigned char *in, unsigned long inlen, ecc_key *key, const ltc_ecc_set_type *dp) { + unsigned long key_size; + unsigned char flags[1]; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(key != NULL); + LTC_ARGCHK(ltc_mp.name != NULL); + + /* init key */ + if (mp_init_multi(&key->pubkey.x, &key->pubkey.y, &key->pubkey.z, &key->k, NULL) != CRYPT_OK) { + return CRYPT_MEM; + } + + /* find out what type of key it is */ + if ((err = der_decode_sequence_multi(in, inlen, + LTC_ASN1_BIT_STRING, 1UL, &flags, + LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { + goto done; + } + + + if (flags[0] == 1) { + /* private key */ + key->type = PK_PRIVATE; + if ((err = der_decode_sequence_multi(in, inlen, + LTC_ASN1_BIT_STRING, 1UL, flags, + LTC_ASN1_SHORT_INTEGER, 1UL, &key_size, + LTC_ASN1_INTEGER, 1UL, key->pubkey.x, + LTC_ASN1_INTEGER, 1UL, key->pubkey.y, + LTC_ASN1_INTEGER, 1UL, key->k, + LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { + goto done; + } + } else { + /* public key */ + key->type = PK_PUBLIC; + if ((err = der_decode_sequence_multi(in, inlen, + LTC_ASN1_BIT_STRING, 1UL, flags, + LTC_ASN1_SHORT_INTEGER, 1UL, &key_size, + LTC_ASN1_INTEGER, 1UL, key->pubkey.x, + LTC_ASN1_INTEGER, 1UL, key->pubkey.y, + LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { + goto done; + } + } + + if (dp == NULL) { + /* find the idx */ + for (key->idx = 0; ltc_ecc_sets[key->idx].size && (unsigned long)ltc_ecc_sets[key->idx].size != key_size; ++key->idx); + if (ltc_ecc_sets[key->idx].size == 0) { + err = CRYPT_INVALID_PACKET; + goto done; + } + key->dp = <c_ecc_sets[key->idx]; + } else { + key->idx = -1; + key->dp = dp; + } + /* set z */ + if ((err = mp_set(key->pubkey.z, 1)) != CRYPT_OK) { + goto done; + } + + /* is it a point on the curve? */ + if ((err = is_point(key)) != CRYPT_OK) { + goto done; + } + + /* we're good */ + return CRYPT_OK; +done: + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_import.c,v $ */ +/* $Revision: 1.13 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_make_key.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Make a new ECC key + @param prng An active PRNG state + @param wprng The index of the PRNG you wish to use + @param keysize The keysize for the new key (in octets from 20 to 65 bytes) + @param key [out] Destination of the newly created key + @return CRYPT_OK if successful, upon error all allocated memory will be freed + */ +int ecc_make_key(prng_state *prng, int wprng, int keysize, ecc_key *key) { + int x, err; + + /* find key size */ + for (x = 0; (keysize > ltc_ecc_sets[x].size) && (ltc_ecc_sets[x].size != 0); x++); + keysize = ltc_ecc_sets[x].size; + + if ((keysize > ECC_MAXSIZE) || (ltc_ecc_sets[x].size == 0)) { + return CRYPT_INVALID_KEYSIZE; + } + err = ecc_make_key_ex(prng, wprng, key, <c_ecc_sets[x]); + key->idx = x; + return err; +} + +int ecc_make_key_ex(prng_state *prng, int wprng, ecc_key *key, const ltc_ecc_set_type *dp) { + int err; + ecc_point *base; + void *prime, *order; + unsigned char *buf; + int keysize; + + LTC_ARGCHK(key != NULL); + LTC_ARGCHK(ltc_mp.name != NULL); + LTC_ARGCHK(dp != NULL); + + /* good prng? */ + if ((err = prng_is_valid(wprng)) != CRYPT_OK) { + return err; + } + + key->idx = -1; + key->dp = dp; + keysize = dp->size; + + /* allocate ram */ + base = NULL; + buf = XMALLOC(ECC_MAXSIZE); + if (buf == NULL) { + return CRYPT_MEM; + } + + /* make up random string */ + if (prng_descriptor[wprng].read(buf, (unsigned long)keysize, prng) != (unsigned long)keysize) { + err = CRYPT_ERROR_READPRNG; + goto ERR_BUF; + } + + /* setup the key variables */ + if ((err = mp_init_multi(&key->pubkey.x, &key->pubkey.y, &key->pubkey.z, &key->k, &prime, &order, NULL)) != CRYPT_OK) { + goto ERR_BUF; + } + base = ltc_ecc_new_point(); + if (base == NULL) { + err = CRYPT_MEM; + goto errkey; + } + + /* read in the specs for this key */ + if ((err = mp_read_radix(prime, (char *)key->dp->prime, 16)) != CRYPT_OK) { + goto errkey; + } + if ((err = mp_read_radix(order, (char *)key->dp->order, 16)) != CRYPT_OK) { + goto errkey; + } + if ((err = mp_read_radix(base->x, (char *)key->dp->Gx, 16)) != CRYPT_OK) { + goto errkey; + } + if ((err = mp_read_radix(base->y, (char *)key->dp->Gy, 16)) != CRYPT_OK) { + goto errkey; + } + if ((err = mp_set(base->z, 1)) != CRYPT_OK) { + goto errkey; + } + if ((err = mp_read_unsigned_bin(key->k, (unsigned char *)buf, keysize)) != CRYPT_OK) { + goto errkey; + } + + /* the key should be smaller than the order of base point */ + if (mp_cmp(key->k, order) != LTC_MP_LT) { + if ((err = mp_mod(key->k, order, key->k)) != CRYPT_OK) { + goto errkey; + } + } + /* make the public key */ + if ((err = ltc_mp.ecc_ptmul(key->k, base, &key->pubkey, prime, 1)) != CRYPT_OK) { + goto errkey; + } + key->type = PK_PRIVATE; + + /* free up ram */ + err = CRYPT_OK; + goto cleanup; +errkey: + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); +cleanup: + ltc_ecc_del_point(base); + mp_clear_multi(prime, order, NULL); +ERR_BUF: + #ifdef LTC_CLEAN_STACK + zeromem(buf, ECC_MAXSIZE); + #endif + XFREE(buf); + return err; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_make_key.c,v $ */ +/* $Revision: 1.13 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_shared_secret.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Create an ECC shared secret between two keys + @param private_key The private ECC key + @param public_key The public key + @param out [out] Destination of the shared secret (Conforms to EC-DH from ANSI X9.63) + @param outlen [in/out] The max size and resulting size of the shared secret + @return CRYPT_OK if successful + */ +int ecc_shared_secret(ecc_key *private_key, ecc_key *public_key, + unsigned char *out, unsigned long *outlen) { + unsigned long x; + ecc_point *result; + void *prime; + int err; + + LTC_ARGCHK(private_key != NULL); + LTC_ARGCHK(public_key != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* type valid? */ + if (private_key->type != PK_PRIVATE) { + return CRYPT_PK_NOT_PRIVATE; + } + + if ((ltc_ecc_is_valid_idx(private_key->idx) == 0) || (ltc_ecc_is_valid_idx(public_key->idx) == 0)) { + return CRYPT_INVALID_ARG; + } + + if (XSTRCMP(private_key->dp->name, public_key->dp->name) != 0) { + return CRYPT_PK_TYPE_MISMATCH; + } + + /* make new point */ + result = ltc_ecc_new_point(); + if (result == NULL) { + return CRYPT_MEM; + } + + if ((err = mp_init(&prime)) != CRYPT_OK) { + ltc_ecc_del_point(result); + return err; + } + + if ((err = mp_read_radix(prime, (char *)private_key->dp->prime, 16)) != CRYPT_OK) { + goto done; + } + if ((err = ltc_mp.ecc_ptmul(private_key->k, &public_key->pubkey, result, prime, 1)) != CRYPT_OK) { + goto done; + } + + x = (unsigned long)mp_unsigned_bin_size(prime); + if (*outlen < x) { + *outlen = x; + err = CRYPT_BUFFER_OVERFLOW; + goto done; + } + zeromem(out, x); + if ((err = mp_to_unsigned_bin(result->x, out + (x - mp_unsigned_bin_size(result->x)))) != CRYPT_OK) { + goto done; + } + + err = CRYPT_OK; + *outlen = x; +done: + mp_clear(prime); + ltc_ecc_del_point(result); + return err; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_shared_secret.c,v $ */ +/* $Revision: 1.10 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_sign_hash.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Sign a message digest + @param in The message digest to sign + @param inlen The length of the digest + @param out [out] The destination for the signature + @param outlen [in/out] The max size and resulting size of the signature + @param prng An active PRNG state + @param wprng The index of the PRNG you wish to use + @param key A private ECC key + @return CRYPT_OK if successful + */ +int ecc_sign_hash(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + prng_state *prng, int wprng, ecc_key *key) { + ecc_key pubkey; + void *r, *s, *e, *p; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(key != NULL); + + /* is this a private key? */ + if (key->type != PK_PRIVATE) { + return CRYPT_PK_NOT_PRIVATE; + } + + /* is the IDX valid ? */ + if (ltc_ecc_is_valid_idx(key->idx) != 1) { + return CRYPT_PK_INVALID_TYPE; + } + + if ((err = prng_is_valid(wprng)) != CRYPT_OK) { + return err; + } + + /* get the hash and load it as a bignum into 'e' */ + /* init the bignums */ + if ((err = mp_init_multi(&r, &s, &p, &e, NULL)) != CRYPT_OK) { + return err; + } + if ((err = mp_read_radix(p, (char *)key->dp->order, 16)) != CRYPT_OK) { + goto errnokey; + } + if ((err = mp_read_unsigned_bin(e, (unsigned char *)in, (int)inlen)) != CRYPT_OK) { + goto errnokey; + } + + /* make up a key and export the public copy */ + for ( ; ; ) { + if ((err = ecc_make_key_ex(prng, wprng, &pubkey, key->dp)) != CRYPT_OK) { + goto errnokey; + } + + /* find r = x1 mod n */ + if ((err = mp_mod(pubkey.pubkey.x, p, r)) != CRYPT_OK) { + goto error; + } + + if (mp_iszero(r) == LTC_MP_YES) { + ecc_free(&pubkey); + } else { + /* find s = (e + xr)/k */ + if ((err = mp_invmod(pubkey.k, p, pubkey.k)) != CRYPT_OK) { + goto error; + } /* k = 1/k */ + if ((err = mp_mulmod(key->k, r, p, s)) != CRYPT_OK) { + goto error; + } /* s = xr */ + if ((err = mp_add(e, s, s)) != CRYPT_OK) { + goto error; + } /* s = e + xr */ + if ((err = mp_mod(s, p, s)) != CRYPT_OK) { + goto error; + } /* s = e + xr */ + if ((err = mp_mulmod(s, pubkey.k, p, s)) != CRYPT_OK) { + goto error; + } /* s = (e + xr)/k */ + ecc_free(&pubkey); + if (mp_iszero(s) == LTC_MP_NO) { + break; + } + } + } + + /* store as SEQUENCE { r, s -- integer } */ + err = der_encode_sequence_multi(out, outlen, + LTC_ASN1_INTEGER, 1UL, r, + LTC_ASN1_INTEGER, 1UL, s, + LTC_ASN1_EOL, 0UL, NULL); + goto errnokey; +error: + ecc_free(&pubkey); +errnokey: + mp_clear_multi(r, s, p, e, NULL); + return err; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_sign_hash.c,v $ */ +/* $Revision: 1.11 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_sizes.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +void ecc_sizes(int *low, int *high) { + int i; + + LTC_ARGCHKVD(low != NULL); + LTC_ARGCHKVD(high != NULL); + + *low = INT_MAX; + *high = 0; + for (i = 0; ltc_ecc_sets[i].size != 0; i++) { + if (ltc_ecc_sets[i].size < *low) { + *low = ltc_ecc_sets[i].size; + } + if (ltc_ecc_sets[i].size > *high) { + *high = ltc_ecc_sets[i].size; + } + } +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_sizes.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_test.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Perform on the ECC system + @return CRYPT_OK if successful + */ +int ecc_test(void) { + void *modulus, *order; + ecc_point *G, *GG; + int i, err, primality; + + if ((err = mp_init_multi(&modulus, &order, NULL)) != CRYPT_OK) { + return err; + } + + G = ltc_ecc_new_point(); + GG = ltc_ecc_new_point(); + if ((G == NULL) || (GG == NULL)) { + mp_clear_multi(modulus, order, NULL); + ltc_ecc_del_point(G); + ltc_ecc_del_point(GG); + return CRYPT_MEM; + } + + for (i = 0; ltc_ecc_sets[i].size; i++) { + #if 0 + printf("Testing %d\n", ltc_ecc_sets[i].size); + #endif + if ((err = mp_read_radix(modulus, (char *)ltc_ecc_sets[i].prime, 16)) != CRYPT_OK) { + goto done; + } + if ((err = mp_read_radix(order, (char *)ltc_ecc_sets[i].order, 16)) != CRYPT_OK) { + goto done; + } + + /* is prime actually prime? */ + if ((err = mp_prime_is_prime(modulus, 8, &primality)) != CRYPT_OK) { + goto done; + } + if (primality == 0) { + err = CRYPT_FAIL_TESTVECTOR; + goto done; + } + + /* is order prime ? */ + if ((err = mp_prime_is_prime(order, 8, &primality)) != CRYPT_OK) { + goto done; + } + if (primality == 0) { + err = CRYPT_FAIL_TESTVECTOR; + goto done; + } + + if ((err = mp_read_radix(G->x, (char *)ltc_ecc_sets[i].Gx, 16)) != CRYPT_OK) { + goto done; + } + if ((err = mp_read_radix(G->y, (char *)ltc_ecc_sets[i].Gy, 16)) != CRYPT_OK) { + goto done; + } + mp_set(G->z, 1); + + /* then we should have G == (order + 1)G */ + if ((err = mp_add_d(order, 1, order)) != CRYPT_OK) { + goto done; + } + if ((err = ltc_mp.ecc_ptmul(order, G, GG, modulus, 1)) != CRYPT_OK) { + goto done; + } + if ((mp_cmp(G->x, GG->x) != LTC_MP_EQ) || (mp_cmp(G->y, GG->y) != LTC_MP_EQ)) { + err = CRYPT_FAIL_TESTVECTOR; + goto done; + } + } + err = CRYPT_OK; +done: + ltc_ecc_del_point(GG); + ltc_ecc_del_point(G); + mp_clear_multi(order, modulus, NULL); + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_test.c,v $ */ +/* $Revision: 1.12 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ecc_verify_hash.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/* verify + * + * w = s^-1 mod n + * u1 = xw + * u2 = rw + * X = u1*G + u2*Q + * v = X_x1 mod n + * accept if v == r + */ + +/** + Verify an ECC signature + @param sig The signature to verify + @param siglen The length of the signature (octets) + @param hash The hash (message digest) that was signed + @param hashlen The length of the hash (octets) + @param stat Result of signature, 1==valid, 0==invalid + @param key The corresponding public ECC key + @return CRYPT_OK if successful (even if the signature is not valid) + */ +int ecc_verify_hash(const unsigned char *sig, unsigned long siglen, + const unsigned char *hash, unsigned long hashlen, + int *stat, ecc_key *key) { + ecc_point *mG, *mQ; + void *r, *s, *v, *w, *u1, *u2, *e, *p, *m; + void *mp; + int err; + + LTC_ARGCHK(sig != NULL); + LTC_ARGCHK(hash != NULL); + LTC_ARGCHK(stat != NULL); + LTC_ARGCHK(key != NULL); + + /* default to invalid signature */ + *stat = 0; + mp = NULL; + + /* is the IDX valid ? */ + if (ltc_ecc_is_valid_idx(key->idx) != 1) { + return CRYPT_PK_INVALID_TYPE; + } + + /* allocate ints */ + if ((err = mp_init_multi(&r, &s, &v, &w, &u1, &u2, &p, &e, &m, NULL)) != CRYPT_OK) { + return CRYPT_MEM; + } + + /* allocate points */ + mG = ltc_ecc_new_point(); + mQ = ltc_ecc_new_point(); + if ((mQ == NULL) || (mG == NULL)) { + err = CRYPT_MEM; + goto error; + } + + /* parse header */ + if ((err = der_decode_sequence_multi(sig, siglen, + LTC_ASN1_INTEGER, 1UL, r, + LTC_ASN1_INTEGER, 1UL, s, + LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { + goto error; + } + + /* get the order */ + if ((err = mp_read_radix(p, (char *)key->dp->order, 16)) != CRYPT_OK) { + goto error; + } + + /* get the modulus */ + if ((err = mp_read_radix(m, (char *)key->dp->prime, 16)) != CRYPT_OK) { + goto error; + } + + /* check for zero */ + if (mp_iszero(r) || mp_iszero(s) || (mp_cmp(r, p) != LTC_MP_LT) || (mp_cmp(s, p) != LTC_MP_LT)) { + err = CRYPT_INVALID_PACKET; + goto error; + } + + /* read hash */ + if ((err = mp_read_unsigned_bin(e, (unsigned char *)hash, (int)hashlen)) != CRYPT_OK) { + goto error; + } + + /* w = s^-1 mod n */ + if ((err = mp_invmod(s, p, w)) != CRYPT_OK) { + goto error; + } + + /* u1 = ew */ + if ((err = mp_mulmod(e, w, p, u1)) != CRYPT_OK) { + goto error; + } + + /* u2 = rw */ + if ((err = mp_mulmod(r, w, p, u2)) != CRYPT_OK) { + goto error; + } + + /* find mG and mQ */ + if ((err = mp_read_radix(mG->x, (char *)key->dp->Gx, 16)) != CRYPT_OK) { + goto error; + } + if ((err = mp_read_radix(mG->y, (char *)key->dp->Gy, 16)) != CRYPT_OK) { + goto error; + } + if ((err = mp_set(mG->z, 1)) != CRYPT_OK) { + goto error; + } + + if ((err = mp_copy(key->pubkey.x, mQ->x)) != CRYPT_OK) { + goto error; + } + if ((err = mp_copy(key->pubkey.y, mQ->y)) != CRYPT_OK) { + goto error; + } + if ((err = mp_copy(key->pubkey.z, mQ->z)) != CRYPT_OK) { + goto error; + } + + /* compute u1*mG + u2*mQ = mG */ + if (ltc_mp.ecc_mul2add == NULL) { + if ((err = ltc_mp.ecc_ptmul(u1, mG, mG, m, 0)) != CRYPT_OK) { + goto error; + } + if ((err = ltc_mp.ecc_ptmul(u2, mQ, mQ, m, 0)) != CRYPT_OK) { + goto error; + } + + /* find the montgomery mp */ + if ((err = mp_montgomery_setup(m, &mp)) != CRYPT_OK) { + goto error; + } + + /* add them */ + if ((err = ltc_mp.ecc_ptadd(mQ, mG, mG, m, mp)) != CRYPT_OK) { + goto error; + } + + /* reduce */ + if ((err = ltc_mp.ecc_map(mG, m, mp)) != CRYPT_OK) { + goto error; + } + } else { + /* use Shamir's trick to compute u1*mG + u2*mQ using half of the doubles */ + if ((err = ltc_mp.ecc_mul2add(mG, u1, mQ, u2, mG, m)) != CRYPT_OK) { + goto error; + } + } + + /* v = X_x1 mod n */ + if ((err = mp_mod(mG->x, p, v)) != CRYPT_OK) { + goto error; + } + + /* does v == r */ + if (mp_cmp(v, r) == LTC_MP_EQ) { + *stat = 1; + } + + /* clear up and return */ + err = CRYPT_OK; +error: + ltc_ecc_del_point(mG); + ltc_ecc_del_point(mQ); + mp_clear_multi(r, s, v, w, u1, u2, p, e, m, NULL); + if (mp != NULL) { + mp_montgomery_free(mp); + } + return err; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_verify_hash.c,v $ */ +/* $Revision: 1.14 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + + +/** + @file error_to_string.c + Convert error codes to ASCII strings, Tom St Denis + */ + +static const char * const err_2_str[] = +{ + "CRYPT_OK", + "CRYPT_ERROR", + "Non-fatal 'no-operation' requested.", + + "Invalid keysize for block cipher.", + "Invalid number of rounds for block cipher.", + "Algorithm failed test vectors.", + + "Buffer overflow.", + "Invalid input packet.", + + "Invalid number of bits for a PRNG.", + "Error reading the PRNG.", + + "Invalid cipher specified.", + "Invalid hash specified.", + "Invalid PRNG specified.", + + "Out of memory.", + + "Invalid PK key or key type specified for function.", + "A private PK key is required.", + + "Invalid argument provided.", + "File Not Found", + + "Invalid PK type.", + "Invalid PK system.", + "Duplicate PK key found on keyring.", + "Key not found in keyring.", + "Invalid sized parameter.", + + "Invalid size for prime.", +}; + +/** + Convert an LTC error code to ASCII + @param err The error code + @return A pointer to the ASCII NUL terminated string for the error or "Invalid error code." if the err code was not valid. + */ +const char *error_to_string(int err) { + if ((err < 0) || (err >= (int)(sizeof(err_2_str) / sizeof(err_2_str[0])))) { + return "Invalid error code."; + } else { + return err_2_str[err]; + } +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/error_to_string.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +#define DESC_DEF_ONLY + + + +/* $Source: /cvs/libtom/libtomcrypt/src/math/gmp_desc.c,v $ */ +/* $Revision: 1.16 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file hash_file.c + Hash a file, Tom St Denis + */ + +/** + @param hash The index of the hash desired + @param fname The name of the file you wish to hash + @param out [out] The destination of the digest + @param outlen [in/out] The max size and resulting size of the message digest + @result CRYPT_OK if successful + */ +int hash_file(int hash, const char *fname, unsigned char *out, unsigned long *outlen) { +#ifdef LTC_NO_FILE + return CRYPT_NOP; +#else + FILE *in; + int err; + LTC_ARGCHK(fname != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + if ((err = hash_is_valid(hash)) != CRYPT_OK) { + return err; + } + + in = fopen(fname, "rb"); + if (in == NULL) { + return CRYPT_FILE_NOTFOUND; + } + + err = hash_filehandle(hash, in, out, outlen); + if (fclose(in) != 0) { + return CRYPT_ERROR; + } + + return err; +#endif +} + +/* $Source: /cvs/libtom/libtomcrypt/src/hashes/helper/hash_file.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:23 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file hash_filehandle.c + Hash open files, Tom St Denis + */ + +/** + Hash data from an open file handle. + @param hash The index of the hash you want to use + @param in The FILE* handle of the file you want to hash + @param out [out] The destination of the digest + @param outlen [in/out] The max size and resulting size of the digest + @result CRYPT_OK if successful + */ +int hash_filehandle(int hash, FILE *in, unsigned char *out, unsigned long *outlen) { +#ifdef LTC_NO_FILE + return CRYPT_NOP; +#else + hash_state md; + unsigned char buf[512]; + size_t x; + int err; + + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(in != NULL); + + if ((err = hash_is_valid(hash)) != CRYPT_OK) { + return err; + } + + if (*outlen < hash_descriptor[hash].hashsize) { + *outlen = hash_descriptor[hash].hashsize; + return CRYPT_BUFFER_OVERFLOW; + } + if ((err = hash_descriptor[hash].init(&md)) != CRYPT_OK) { + return err; + } + + *outlen = hash_descriptor[hash].hashsize; + do { + x = fread(buf, 1, sizeof(buf), in); + if ((err = hash_descriptor[hash].process(&md, buf, x)) != CRYPT_OK) { + return err; + } + } while (x == sizeof(buf)); + err = hash_descriptor[hash].done(&md, out); + + #ifdef LTC_CLEAN_STACK + zeromem(buf, sizeof(buf)); + #endif + return err; +#endif +} + +/* $Source: /cvs/libtom/libtomcrypt/src/hashes/helper/hash_filehandle.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:23 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file hash_memory.c + Hash memory helper, Tom St Denis + */ + +/** + Hash a block of memory and store the digest. + @param hash The index of the hash you wish to use + @param in The data you wish to hash + @param inlen The length of the data to hash (octets) + @param out [out] Where to store the digest + @param outlen [in/out] Max size and resulting size of the digest + @return CRYPT_OK if successful + */ +int hash_memory(int hash, const unsigned char *in, unsigned long inlen, unsigned char *out, unsigned long *outlen) { + hash_state *md; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + if ((err = hash_is_valid(hash)) != CRYPT_OK) { + return err; + } + + if (*outlen < hash_descriptor[hash].hashsize) { + *outlen = hash_descriptor[hash].hashsize; + return CRYPT_BUFFER_OVERFLOW; + } + + md = XMALLOC(sizeof(hash_state)); + if (md == NULL) { + return CRYPT_MEM; + } + + if ((err = hash_descriptor[hash].init(md)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash].process(md, in, inlen)) != CRYPT_OK) { + goto LBL_ERR; + } + err = hash_descriptor[hash].done(md, out); + *outlen = hash_descriptor[hash].hashsize; +LBL_ERR: +#ifdef LTC_CLEAN_STACK + zeromem(md, sizeof(hash_state)); +#endif + XFREE(md); + + return err; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/hashes/helper/hash_memory.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:23 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +#include + +/** + @file hash_memory_multi.c + Hash (multiple buffers) memory helper, Tom St Denis + */ + +/** + Hash multiple (non-adjacent) blocks of memory at once. + @param hash The index of the hash you wish to use + @param out [out] Where to store the digest + @param outlen [in/out] Max size and resulting size of the digest + @param in The data you wish to hash + @param inlen The length of the data to hash (octets) + @param ... tuples of (data,len) pairs to hash, terminated with a (NULL,x) (x=don't care) + @return CRYPT_OK if successful + */ +int hash_memory_multi(int hash, unsigned char *out, unsigned long *outlen, + const unsigned char *in, unsigned long inlen, ...) { + hash_state *md; + int err; + va_list args; + const unsigned char *curptr; + unsigned long curlen; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + if ((err = hash_is_valid(hash)) != CRYPT_OK) { + return err; + } + + if (*outlen < hash_descriptor[hash].hashsize) { + *outlen = hash_descriptor[hash].hashsize; + return CRYPT_BUFFER_OVERFLOW; + } + + md = XMALLOC(sizeof(hash_state)); + if (md == NULL) { + return CRYPT_MEM; + } + + if ((err = hash_descriptor[hash].init(md)) != CRYPT_OK) { + goto LBL_ERR; + } + + va_start(args, inlen); + curptr = in; + curlen = inlen; + for ( ; ; ) { + /* process buf */ + if ((err = hash_descriptor[hash].process(md, curptr, curlen)) != CRYPT_OK) { + goto LBL_ERR; + } + /* step to next */ + curptr = va_arg(args, const unsigned char *); + if (curptr == NULL) { + break; + } + curlen = va_arg(args, unsigned long); + } + err = hash_descriptor[hash].done(md, out); + *outlen = hash_descriptor[hash].hashsize; +LBL_ERR: +#ifdef LTC_CLEAN_STACK + zeromem(md, sizeof(hash_state)); +#endif + XFREE(md); + va_end(args); + return err; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/hashes/helper/hash_memory_multi.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:23 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ltc_ecc_is_valid_idx.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** Returns whether an ECC idx is valid or not + @param n The idx number to check + @return 1 if valid, 0 if not + */ +int ltc_ecc_is_valid_idx(int n) { + int x; + + for (x = 0; ltc_ecc_sets[x].size != 0; x++); + /* -1 is a valid index --- indicating that the domain params were supplied by the user */ + if ((n >= -1) && (n < x)) { + return 1; + } + return 0; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ltc_ecc_is_valid_idx.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ltc_ecc_map.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Map a projective jacbobian point back to affine space + @param P [in/out] The point to map + @param modulus The modulus of the field the ECC curve is in + @param mp The "b" value from montgomery_setup() + @return CRYPT_OK on success + */ +int ltc_ecc_map(ecc_point *P, void *modulus, void *mp) { + void *t1, *t2; + int err; + + LTC_ARGCHK(P != NULL); + LTC_ARGCHK(modulus != NULL); + LTC_ARGCHK(mp != NULL); + + if ((err = mp_init_multi(&t1, &t2, NULL)) != CRYPT_OK) { + return CRYPT_MEM; + } + + /* first map z back to normal */ + if ((err = mp_montgomery_reduce(P->z, modulus, mp)) != CRYPT_OK) { + goto done; + } + + /* get 1/z */ + if ((err = mp_invmod(P->z, modulus, t1)) != CRYPT_OK) { + goto done; + } + + /* get 1/z^2 and 1/z^3 */ + if ((err = mp_sqr(t1, t2)) != CRYPT_OK) { + goto done; + } + if ((err = mp_mod(t2, modulus, t2)) != CRYPT_OK) { + goto done; + } + if ((err = mp_mul(t1, t2, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_mod(t1, modulus, t1)) != CRYPT_OK) { + goto done; + } + + /* multiply against x/y */ + if ((err = mp_mul(P->x, t2, P->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(P->x, modulus, mp)) != CRYPT_OK) { + goto done; + } + if ((err = mp_mul(P->y, t1, P->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(P->y, modulus, mp)) != CRYPT_OK) { + goto done; + } + if ((err = mp_set(P->z, 1)) != CRYPT_OK) { + goto done; + } + + err = CRYPT_OK; +done: + mp_clear_multi(t1, t2, NULL); + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ltc_ecc_map.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ltc_ecc_mul2add.c + ECC Crypto, Shamir's Trick, Tom St Denis + */ + +#ifdef LTC_MECC + + #ifdef LTC_ECC_SHAMIR + +/** Computes kA*A + kB*B = C using Shamir's Trick + @param A First point to multiply + @param kA What to multiple A by + @param B Second point to multiply + @param kB What to multiple B by + @param C [out] Destination point (can overlap with A or B + @param modulus Modulus for curve + @return CRYPT_OK on success + */ +int ltc_ecc_mul2add(ecc_point *A, void *kA, + ecc_point *B, void *kB, + ecc_point *C, + void *modulus) { + ecc_point *precomp[16]; + unsigned bitbufA, bitbufB, lenA, lenB, len, x, y, nA, nB, nibble; + unsigned char *tA, *tB; + int err, first; + void *mp, *mu; + + /* argchks */ + LTC_ARGCHK(A != NULL); + LTC_ARGCHK(B != NULL); + LTC_ARGCHK(C != NULL); + LTC_ARGCHK(kA != NULL); + LTC_ARGCHK(kB != NULL); + LTC_ARGCHK(modulus != NULL); + + /* allocate memory */ + tA = XCALLOC(1, ECC_BUF_SIZE); + if (tA == NULL) { + return CRYPT_MEM; + } + tB = XCALLOC(1, ECC_BUF_SIZE); + if (tB == NULL) { + XFREE(tA); + return CRYPT_MEM; + } + + /* get sizes */ + lenA = mp_unsigned_bin_size(kA); + lenB = mp_unsigned_bin_size(kB); + len = MAX(lenA, lenB); + + /* sanity check */ + if ((lenA > ECC_BUF_SIZE) || (lenB > ECC_BUF_SIZE)) { + err = CRYPT_INVALID_ARG; + goto ERR_T; + } + + /* extract and justify kA */ + mp_to_unsigned_bin(kA, (len - lenA) + tA); + + /* extract and justify kB */ + mp_to_unsigned_bin(kB, (len - lenB) + tB); + + /* allocate the table */ + for (x = 0; x < 16; x++) { + precomp[x] = ltc_ecc_new_point(); + if (precomp[x] == NULL) { + for (y = 0; y < x; ++y) { + ltc_ecc_del_point(precomp[y]); + } + err = CRYPT_MEM; + goto ERR_T; + } + } + + /* init montgomery reduction */ + if ((err = mp_montgomery_setup(modulus, &mp)) != CRYPT_OK) { + goto ERR_P; + } + if ((err = mp_init(&mu)) != CRYPT_OK) { + goto ERR_MP; + } + if ((err = mp_montgomery_normalization(mu, modulus)) != CRYPT_OK) { + goto ERR_MU; + } + + /* copy ones ... */ + if ((err = mp_mulmod(A->x, mu, modulus, precomp[1]->x)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = mp_mulmod(A->y, mu, modulus, precomp[1]->y)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = mp_mulmod(A->z, mu, modulus, precomp[1]->z)) != CRYPT_OK) { + goto ERR_MU; + } + + if ((err = mp_mulmod(B->x, mu, modulus, precomp[1 << 2]->x)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = mp_mulmod(B->y, mu, modulus, precomp[1 << 2]->y)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = mp_mulmod(B->z, mu, modulus, precomp[1 << 2]->z)) != CRYPT_OK) { + goto ERR_MU; + } + + /* precomp [i,0](A + B) table */ + if ((err = ltc_mp.ecc_ptdbl(precomp[1], precomp[2], modulus, mp)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = ltc_mp.ecc_ptadd(precomp[1], precomp[2], precomp[3], modulus, mp)) != CRYPT_OK) { + goto ERR_MU; + } + + /* precomp [0,i](A + B) table */ + if ((err = ltc_mp.ecc_ptdbl(precomp[1 << 2], precomp[2 << 2], modulus, mp)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = ltc_mp.ecc_ptadd(precomp[1 << 2], precomp[2 << 2], precomp[3 << 2], modulus, mp)) != CRYPT_OK) { + goto ERR_MU; + } + + /* precomp [i,j](A + B) table (i != 0, j != 0) */ + for (x = 1; x < 4; x++) { + for (y = 1; y < 4; y++) { + if ((err = ltc_mp.ecc_ptadd(precomp[x], precomp[(y << 2)], precomp[x + (y << 2)], modulus, mp)) != CRYPT_OK) { + goto ERR_MU; + } + } + } + + nibble = 3; + first = 1; + bitbufA = tA[0]; + bitbufB = tB[0]; + + /* for every byte of the multiplicands */ + for (x = -1; ; ) { + /* grab a nibble */ + if (++nibble == 4) { + ++x; + if (x == len) break; + bitbufA = tA[x]; + bitbufB = tB[x]; + nibble = 0; + } + + /* extract two bits from both, shift/update */ + nA = (bitbufA >> 6) & 0x03; + nB = (bitbufB >> 6) & 0x03; + bitbufA = (bitbufA << 2) & 0xFF; + bitbufB = (bitbufB << 2) & 0xFF; + + /* if both zero, if first, continue */ + if ((nA == 0) && (nB == 0) && (first == 1)) { + continue; + } + + /* double twice, only if this isn't the first */ + if (first == 0) { + /* double twice */ + if ((err = ltc_mp.ecc_ptdbl(C, C, modulus, mp)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = ltc_mp.ecc_ptdbl(C, C, modulus, mp)) != CRYPT_OK) { + goto ERR_MU; + } + } + + /* if not both zero */ + if ((nA != 0) || (nB != 0)) { + if (first == 1) { + /* if first, copy from table */ + first = 0; + if ((err = mp_copy(precomp[nA + (nB << 2)]->x, C->x)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = mp_copy(precomp[nA + (nB << 2)]->y, C->y)) != CRYPT_OK) { + goto ERR_MU; + } + if ((err = mp_copy(precomp[nA + (nB << 2)]->z, C->z)) != CRYPT_OK) { + goto ERR_MU; + } + } else { + /* if not first, add from table */ + if ((err = ltc_mp.ecc_ptadd(C, precomp[nA + (nB << 2)], C, modulus, mp)) != CRYPT_OK) { + goto ERR_MU; + } + } + } + } + + /* reduce to affine */ + err = ltc_ecc_map(C, modulus, mp); + + /* clean up */ +ERR_MU: + mp_clear(mu); +ERR_MP: + mp_montgomery_free(mp); +ERR_P: + for (x = 0; x < 16; x++) { + ltc_ecc_del_point(precomp[x]); + } +ERR_T: + #ifdef LTC_CLEAN_STACK + zeromem(tA, ECC_BUF_SIZE); + zeromem(tB, ECC_BUF_SIZE); + #endif + XFREE(tA); + XFREE(tB); + + return err; +} + #endif +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ltc_ecc_mul2add.c,v $ */ +/* $Revision: 1.8 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ltc_ecc_mulmod.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + #ifndef LTC_ECC_TIMING_RESISTANT + +/* size of sliding window, don't change this! */ + #define WINSIZE 4 + +/** + Perform a point multiplication + @param k The scalar to multiply by + @param G The base point + @param R [out] Destination for kG + @param modulus The modulus of the field the ECC curve is in + @param map Boolean whether to map back to affine or not (1==map, 0 == leave in projective) + @return CRYPT_OK on success + */ +int ltc_ecc_mulmod(void *k, ecc_point *G, ecc_point *R, void *modulus, int map) { + ecc_point *tG, *M[8]; + int i, j, err; + void *mu, *mp; + unsigned long buf; + int first, bitbuf, bitcpy, bitcnt, mode, digidx; + + LTC_ARGCHK(k != NULL); + LTC_ARGCHK(G != NULL); + LTC_ARGCHK(R != NULL); + LTC_ARGCHK(modulus != NULL); + + /* init montgomery reduction */ + if ((err = mp_montgomery_setup(modulus, &mp)) != CRYPT_OK) { + return err; + } + if ((err = mp_init(&mu)) != CRYPT_OK) { + mp_montgomery_free(mp); + return err; + } + if ((err = mp_montgomery_normalization(mu, modulus)) != CRYPT_OK) { + mp_montgomery_free(mp); + mp_clear(mu); + return err; + } + + /* alloc ram for window temps */ + for (i = 0; i < 8; i++) { + M[i] = ltc_ecc_new_point(); + if (M[i] == NULL) { + for (j = 0; j < i; j++) { + ltc_ecc_del_point(M[j]); + } + mp_montgomery_free(mp); + mp_clear(mu); + return CRYPT_MEM; + } + } + + /* make a copy of G incase R==G */ + tG = ltc_ecc_new_point(); + if (tG == NULL) { + err = CRYPT_MEM; + goto done; + } + + /* tG = G and convert to montgomery */ + if (mp_cmp_d(mu, 1) == LTC_MP_EQ) { + if ((err = mp_copy(G->x, tG->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(G->y, tG->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(G->z, tG->z)) != CRYPT_OK) { + goto done; + } + } else { + if ((err = mp_mulmod(G->x, mu, modulus, tG->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_mulmod(G->y, mu, modulus, tG->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_mulmod(G->z, mu, modulus, tG->z)) != CRYPT_OK) { + goto done; + } + } + mp_clear(mu); + mu = NULL; + + /* calc the M tab, which holds kG for k==8..15 */ + /* M[0] == 8G */ + if ((err = ltc_mp.ecc_ptdbl(tG, M[0], modulus, mp)) != CRYPT_OK) { + goto done; + } + if ((err = ltc_mp.ecc_ptdbl(M[0], M[0], modulus, mp)) != CRYPT_OK) { + goto done; + } + if ((err = ltc_mp.ecc_ptdbl(M[0], M[0], modulus, mp)) != CRYPT_OK) { + goto done; + } + + /* now find (8+k)G for k=1..7 */ + for (j = 9; j < 16; j++) { + if ((err = ltc_mp.ecc_ptadd(M[j - 9], tG, M[j - 8], modulus, mp)) != CRYPT_OK) { + goto done; + } + } + + /* setup sliding window */ + mode = 0; + bitcnt = 1; + buf = 0; + digidx = mp_get_digit_count(k) - 1; + bitcpy = bitbuf = 0; + first = 1; + + /* perform ops */ + for ( ; ; ) { + /* grab next digit as required */ + if (--bitcnt == 0) { + if (digidx == -1) { + break; + } + buf = mp_get_digit(k, digidx); + bitcnt = (int)ltc_mp.bits_per_digit; + --digidx; + } + + /* grab the next msb from the ltiplicand */ + i = (buf >> (ltc_mp.bits_per_digit - 1)) & 1; + buf <<= 1; + + /* skip leading zero bits */ + if ((mode == 0) && (i == 0)) { + continue; + } + + /* if the bit is zero and mode == 1 then we double */ + if ((mode == 1) && (i == 0)) { + if ((err = ltc_mp.ecc_ptdbl(R, R, modulus, mp)) != CRYPT_OK) { + goto done; + } + continue; + } + + /* else we add it to the window */ + bitbuf |= (i << (WINSIZE - ++bitcpy)); + mode = 2; + + if (bitcpy == WINSIZE) { + /* if this is the first window we do a simple copy */ + if (first == 1) { + /* R = kG [k = first window] */ + if ((err = mp_copy(M[bitbuf - 8]->x, R->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(M[bitbuf - 8]->y, R->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(M[bitbuf - 8]->z, R->z)) != CRYPT_OK) { + goto done; + } + first = 0; + } else { + /* normal window */ + /* ok window is filled so double as required and add */ + /* double first */ + for (j = 0; j < WINSIZE; j++) { + if ((err = ltc_mp.ecc_ptdbl(R, R, modulus, mp)) != CRYPT_OK) { + goto done; + } + } + + /* then add, bitbuf will be 8..15 [8..2^WINSIZE] guaranteed */ + if ((err = ltc_mp.ecc_ptadd(R, M[bitbuf - 8], R, modulus, mp)) != CRYPT_OK) { + goto done; + } + } + /* empty window and reset */ + bitcpy = bitbuf = 0; + mode = 1; + } + } + + /* if bits remain then double/add */ + if ((mode == 2) && (bitcpy > 0)) { + /* double then add */ + for (j = 0; j < bitcpy; j++) { + /* only double if we have had at least one add first */ + if (first == 0) { + if ((err = ltc_mp.ecc_ptdbl(R, R, modulus, mp)) != CRYPT_OK) { + goto done; + } + } + + bitbuf <<= 1; + if ((bitbuf & (1 << WINSIZE)) != 0) { + if (first == 1) { + /* first add, so copy */ + if ((err = mp_copy(tG->x, R->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(tG->y, R->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(tG->z, R->z)) != CRYPT_OK) { + goto done; + } + first = 0; + } else { + /* then add */ + if ((err = ltc_mp.ecc_ptadd(R, tG, R, modulus, mp)) != CRYPT_OK) { + goto done; + } + } + } + } + } + + /* map R back from projective space */ + if (map) { + err = ltc_ecc_map(R, modulus, mp); + } else { + err = CRYPT_OK; + } +done: + if (mu != NULL) { + mp_clear(mu); + } + mp_montgomery_free(mp); + ltc_ecc_del_point(tG); + for (i = 0; i < 8; i++) { + ltc_ecc_del_point(M[i]); + } + return err; +} + #endif + + #undef WINSIZE +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ltc_ecc_mulmod.c,v $ */ +/* $Revision: 1.26 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ltc_ecc_mulmod_timing.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + + #ifdef LTC_ECC_TIMING_RESISTANT + +/** + Perform a point multiplication (timing resistant) + @param k The scalar to multiply by + @param G The base point + @param R [out] Destination for kG + @param modulus The modulus of the field the ECC curve is in + @param map Boolean whether to map back to affine or not (1==map, 0 == leave in projective) + @return CRYPT_OK on success + */ +int ltc_ecc_mulmod(void *k, ecc_point *G, ecc_point *R, void *modulus, int map) { + ecc_point *tG, *M[3]; + int i, j, err; + void *mu, *mp; + unsigned long buf; + int first, bitbuf, bitcpy, bitcnt, mode, digidx; + + LTC_ARGCHK(k != NULL); + LTC_ARGCHK(G != NULL); + LTC_ARGCHK(R != NULL); + LTC_ARGCHK(modulus != NULL); + + /* init montgomery reduction */ + if ((err = mp_montgomery_setup(modulus, &mp)) != CRYPT_OK) { + return err; + } + if ((err = mp_init(&mu)) != CRYPT_OK) { + mp_montgomery_free(mp); + return err; + } + if ((err = mp_montgomery_normalization(mu, modulus)) != CRYPT_OK) { + mp_clear(mu); + mp_montgomery_free(mp); + return err; + } + + /* alloc ram for window temps */ + for (i = 0; i < 3; i++) { + M[i] = ltc_ecc_new_point(); + if (M[i] == NULL) { + for (j = 0; j < i; j++) { + ltc_ecc_del_point(M[j]); + } + mp_clear(mu); + mp_montgomery_free(mp); + return CRYPT_MEM; + } + } + + /* make a copy of G incase R==G */ + tG = ltc_ecc_new_point(); + if (tG == NULL) { + err = CRYPT_MEM; + goto done; + } + + /* tG = G and convert to montgomery */ + if ((err = mp_mulmod(G->x, mu, modulus, tG->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_mulmod(G->y, mu, modulus, tG->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_mulmod(G->z, mu, modulus, tG->z)) != CRYPT_OK) { + goto done; + } + mp_clear(mu); + mu = NULL; + + /* calc the M tab */ + /* M[0] == G */ + if ((err = mp_copy(tG->x, M[0]->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(tG->y, M[0]->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(tG->z, M[0]->z)) != CRYPT_OK) { + goto done; + } + /* M[1] == 2G */ + if ((err = ltc_mp.ecc_ptdbl(tG, M[1], modulus, mp)) != CRYPT_OK) { + goto done; + } + + /* setup sliding window */ + mode = 0; + bitcnt = 1; + buf = 0; + digidx = mp_get_digit_count(k) - 1; + bitcpy = bitbuf = 0; + first = 1; + + /* perform ops */ + for ( ; ; ) { + /* grab next digit as required */ + if (--bitcnt == 0) { + if (digidx == -1) { + break; + } + buf = mp_get_digit(k, digidx); + bitcnt = (int)MP_DIGIT_BIT; + --digidx; + } + + /* grab the next msb from the ltiplicand */ + i = (buf >> (MP_DIGIT_BIT - 1)) & 1; + buf <<= 1; + + if ((mode == 0) && (i == 0)) { + /* dummy operations */ + if ((err = ltc_mp.ecc_ptadd(M[0], M[1], M[2], modulus, mp)) != CRYPT_OK) { + goto done; + } + if ((err = ltc_mp.ecc_ptdbl(M[1], M[2], modulus, mp)) != CRYPT_OK) { + goto done; + } + continue; + } + + if ((mode == 0) && (i == 1)) { + mode = 1; + /* dummy operations */ + if ((err = ltc_mp.ecc_ptadd(M[0], M[1], M[2], modulus, mp)) != CRYPT_OK) { + goto done; + } + if ((err = ltc_mp.ecc_ptdbl(M[1], M[2], modulus, mp)) != CRYPT_OK) { + goto done; + } + continue; + } + + if ((err = ltc_mp.ecc_ptadd(M[0], M[1], M[i ^ 1], modulus, mp)) != CRYPT_OK) { + goto done; + } + if ((err = ltc_mp.ecc_ptdbl(M[i], M[i], modulus, mp)) != CRYPT_OK) { + goto done; + } + } + + /* copy result out */ + if ((err = mp_copy(M[0]->x, R->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(M[0]->y, R->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(M[0]->z, R->z)) != CRYPT_OK) { + goto done; + } + + /* map R back from projective space */ + if (map) { + err = ltc_ecc_map(R, modulus, mp); + } else { + err = CRYPT_OK; + } +done: + if (mu != NULL) { + mp_clear(mu); + } + mp_montgomery_free(mp); + ltc_ecc_del_point(tG); + for (i = 0; i < 3; i++) { + ltc_ecc_del_point(M[i]); + } + return err; +} + #endif +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ltc_ecc_mulmod_timing.c,v $ */ +/* $Revision: 1.13 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ltc_ecc_points.c + ECC Crypto, Tom St Denis + */ + +#ifdef LTC_MECC + +/** + Allocate a new ECC point + @return A newly allocated point or NULL on error + */ +ecc_point *ltc_ecc_new_point(void) { + ecc_point *p; + + p = XCALLOC(1, sizeof(*p)); + if (p == NULL) { + return NULL; + } + if (mp_init_multi(&p->x, &p->y, &p->z, NULL) != CRYPT_OK) { + XFREE(p); + return NULL; + } + return p; +} + +/** Free an ECC point from memory + @param p The point to free + */ +void ltc_ecc_del_point(ecc_point *p) { + /* prevents free'ing null arguments */ + if (p != NULL) { + mp_clear_multi(p->x, p->y, p->z, NULL); /* note: p->z may be NULL but that's ok with this function anyways */ + XFREE(p); + } +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ltc_ecc_points.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ltc_ecc_projective_add_point.c + ECC Crypto, Tom St Denis + */ + +#if defined(LTC_MECC) && (!defined(LTC_MECC_ACCEL) || defined(LTM_LTC_DESC)) + +/** + Add two ECC points + @param P The point to add + @param Q The point to add + @param R [out] The destination of the double + @param modulus The modulus of the field the ECC curve is in + @param mp The "b" value from montgomery_setup() + @return CRYPT_OK on success + */ +int ltc_ecc_projective_add_point(ecc_point *P, ecc_point *Q, ecc_point *R, void *modulus, void *mp) { + void *t1, *t2, *x, *y, *z; + int err; + + LTC_ARGCHK(P != NULL); + LTC_ARGCHK(Q != NULL); + LTC_ARGCHK(R != NULL); + LTC_ARGCHK(modulus != NULL); + LTC_ARGCHK(mp != NULL); + + if ((err = mp_init_multi(&t1, &t2, &x, &y, &z, NULL)) != CRYPT_OK) { + return err; + } + + /* should we dbl instead? */ + if ((err = mp_sub(modulus, Q->y, t1)) != CRYPT_OK) { + goto done; + } + + if ((mp_cmp(P->x, Q->x) == LTC_MP_EQ) && + ((Q->z != NULL) && (mp_cmp(P->z, Q->z) == LTC_MP_EQ)) && + ((mp_cmp(P->y, Q->y) == LTC_MP_EQ) || (mp_cmp(P->y, t1) == LTC_MP_EQ))) { + mp_clear_multi(t1, t2, x, y, z, NULL); + return ltc_ecc_projective_dbl_point(P, R, modulus, mp); + } + + if ((err = mp_copy(P->x, x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(P->y, y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(P->z, z)) != CRYPT_OK) { + goto done; + } + + /* if Z is one then these are no-operations */ + if (Q->z != NULL) { + /* T1 = Z' * Z' */ + if ((err = mp_sqr(Q->z, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t1, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* X = X * T1 */ + if ((err = mp_mul(t1, x, x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(x, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T1 = Z' * T1 */ + if ((err = mp_mul(Q->z, t1, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t1, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* Y = Y * T1 */ + if ((err = mp_mul(t1, y, y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(y, modulus, mp)) != CRYPT_OK) { + goto done; + } + } + + /* T1 = Z*Z */ + if ((err = mp_sqr(z, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t1, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T2 = X' * T1 */ + if ((err = mp_mul(Q->x, t1, t2)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t2, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T1 = Z * T1 */ + if ((err = mp_mul(z, t1, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t1, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T1 = Y' * T1 */ + if ((err = mp_mul(Q->y, t1, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t1, modulus, mp)) != CRYPT_OK) { + goto done; + } + + /* Y = Y - T1 */ + if ((err = mp_sub(y, t1, y)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(y, 0) == LTC_MP_LT) { + if ((err = mp_add(y, modulus, y)) != CRYPT_OK) { + goto done; + } + } + /* T1 = 2T1 */ + if ((err = mp_add(t1, t1, t1)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(t1, modulus) != LTC_MP_LT) { + if ((err = mp_sub(t1, modulus, t1)) != CRYPT_OK) { + goto done; + } + } + /* T1 = Y + T1 */ + if ((err = mp_add(t1, y, t1)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(t1, modulus) != LTC_MP_LT) { + if ((err = mp_sub(t1, modulus, t1)) != CRYPT_OK) { + goto done; + } + } + /* X = X - T2 */ + if ((err = mp_sub(x, t2, x)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(x, 0) == LTC_MP_LT) { + if ((err = mp_add(x, modulus, x)) != CRYPT_OK) { + goto done; + } + } + /* T2 = 2T2 */ + if ((err = mp_add(t2, t2, t2)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(t2, modulus) != LTC_MP_LT) { + if ((err = mp_sub(t2, modulus, t2)) != CRYPT_OK) { + goto done; + } + } + /* T2 = X + T2 */ + if ((err = mp_add(t2, x, t2)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(t2, modulus) != LTC_MP_LT) { + if ((err = mp_sub(t2, modulus, t2)) != CRYPT_OK) { + goto done; + } + } + + /* if Z' != 1 */ + if (Q->z != NULL) { + /* Z = Z * Z' */ + if ((err = mp_mul(z, Q->z, z)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(z, modulus, mp)) != CRYPT_OK) { + goto done; + } + } + + /* Z = Z * X */ + if ((err = mp_mul(z, x, z)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(z, modulus, mp)) != CRYPT_OK) { + goto done; + } + + /* T1 = T1 * X */ + if ((err = mp_mul(t1, x, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t1, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* X = X * X */ + if ((err = mp_sqr(x, x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(x, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T2 = T2 * x */ + if ((err = mp_mul(t2, x, t2)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t2, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T1 = T1 * X */ + if ((err = mp_mul(t1, x, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t1, modulus, mp)) != CRYPT_OK) { + goto done; + } + + /* X = Y*Y */ + if ((err = mp_sqr(y, x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(x, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* X = X - T2 */ + if ((err = mp_sub(x, t2, x)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(x, 0) == LTC_MP_LT) { + if ((err = mp_add(x, modulus, x)) != CRYPT_OK) { + goto done; + } + } + + /* T2 = T2 - X */ + if ((err = mp_sub(t2, x, t2)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(t2, 0) == LTC_MP_LT) { + if ((err = mp_add(t2, modulus, t2)) != CRYPT_OK) { + goto done; + } + } + /* T2 = T2 - X */ + if ((err = mp_sub(t2, x, t2)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(t2, 0) == LTC_MP_LT) { + if ((err = mp_add(t2, modulus, t2)) != CRYPT_OK) { + goto done; + } + } + /* T2 = T2 * Y */ + if ((err = mp_mul(t2, y, t2)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t2, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* Y = T2 - T1 */ + if ((err = mp_sub(t2, t1, y)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(y, 0) == LTC_MP_LT) { + if ((err = mp_add(y, modulus, y)) != CRYPT_OK) { + goto done; + } + } + /* Y = Y/2 */ + if (mp_isodd(y)) { + if ((err = mp_add(y, modulus, y)) != CRYPT_OK) { + goto done; + } + } + if ((err = mp_div_2(y, y)) != CRYPT_OK) { + goto done; + } + + if ((err = mp_copy(x, R->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(y, R->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(z, R->z)) != CRYPT_OK) { + goto done; + } + + err = CRYPT_OK; +done: + mp_clear_multi(t1, t2, x, y, z, NULL); + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ltc_ecc_projective_add_point.c,v $ */ +/* $Revision: 1.16 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* Implements ECC over Z/pZ for curve y^2 = x^3 - 3x + b + * + * All curves taken from NIST recommendation paper of July 1999 + * Available at http://csrc.nist.gov/cryptval/dss.htm + */ + + +/** + @file ltc_ecc_projective_dbl_point.c + ECC Crypto, Tom St Denis + */ + +#if defined(LTC_MECC) && (!defined(LTC_MECC_ACCEL) || defined(LTM_LTC_DESC)) + +/** + Double an ECC point + @param P The point to double + @param R [out] The destination of the double + @param modulus The modulus of the field the ECC curve is in + @param mp The "b" value from montgomery_setup() + @return CRYPT_OK on success + */ +int ltc_ecc_projective_dbl_point(ecc_point *P, ecc_point *R, void *modulus, void *mp) { + void *t1, *t2; + int err; + + LTC_ARGCHK(P != NULL); + LTC_ARGCHK(R != NULL); + LTC_ARGCHK(modulus != NULL); + LTC_ARGCHK(mp != NULL); + + if ((err = mp_init_multi(&t1, &t2, NULL)) != CRYPT_OK) { + return err; + } + + if (P != R) { + if ((err = mp_copy(P->x, R->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(P->y, R->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_copy(P->z, R->z)) != CRYPT_OK) { + goto done; + } + } + + /* t1 = Z * Z */ + if ((err = mp_sqr(R->z, t1)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t1, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* Z = Y * Z */ + if ((err = mp_mul(R->z, R->y, R->z)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(R->z, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* Z = 2Z */ + if ((err = mp_add(R->z, R->z, R->z)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(R->z, modulus) != LTC_MP_LT) { + if ((err = mp_sub(R->z, modulus, R->z)) != CRYPT_OK) { + goto done; + } + } + + /* T2 = X - T1 */ + if ((err = mp_sub(R->x, t1, t2)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(t2, 0) == LTC_MP_LT) { + if ((err = mp_add(t2, modulus, t2)) != CRYPT_OK) { + goto done; + } + } + /* T1 = X + T1 */ + if ((err = mp_add(t1, R->x, t1)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(t1, modulus) != LTC_MP_LT) { + if ((err = mp_sub(t1, modulus, t1)) != CRYPT_OK) { + goto done; + } + } + /* T2 = T1 * T2 */ + if ((err = mp_mul(t1, t2, t2)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t2, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T1 = 2T2 */ + if ((err = mp_add(t2, t2, t1)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(t1, modulus) != LTC_MP_LT) { + if ((err = mp_sub(t1, modulus, t1)) != CRYPT_OK) { + goto done; + } + } + /* T1 = T1 + T2 */ + if ((err = mp_add(t1, t2, t1)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(t1, modulus) != LTC_MP_LT) { + if ((err = mp_sub(t1, modulus, t1)) != CRYPT_OK) { + goto done; + } + } + + /* Y = 2Y */ + if ((err = mp_add(R->y, R->y, R->y)) != CRYPT_OK) { + goto done; + } + if (mp_cmp(R->y, modulus) != LTC_MP_LT) { + if ((err = mp_sub(R->y, modulus, R->y)) != CRYPT_OK) { + goto done; + } + } + /* Y = Y * Y */ + if ((err = mp_sqr(R->y, R->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(R->y, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T2 = Y * Y */ + if ((err = mp_sqr(R->y, t2)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(t2, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* T2 = T2/2 */ + if (mp_isodd(t2)) { + if ((err = mp_add(t2, modulus, t2)) != CRYPT_OK) { + goto done; + } + } + if ((err = mp_div_2(t2, t2)) != CRYPT_OK) { + goto done; + } + /* Y = Y * X */ + if ((err = mp_mul(R->y, R->x, R->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(R->y, modulus, mp)) != CRYPT_OK) { + goto done; + } + + /* X = T1 * T1 */ + if ((err = mp_sqr(t1, R->x)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(R->x, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* X = X - Y */ + if ((err = mp_sub(R->x, R->y, R->x)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(R->x, 0) == LTC_MP_LT) { + if ((err = mp_add(R->x, modulus, R->x)) != CRYPT_OK) { + goto done; + } + } + /* X = X - Y */ + if ((err = mp_sub(R->x, R->y, R->x)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(R->x, 0) == LTC_MP_LT) { + if ((err = mp_add(R->x, modulus, R->x)) != CRYPT_OK) { + goto done; + } + } + + /* Y = Y - X */ + if ((err = mp_sub(R->y, R->x, R->y)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(R->y, 0) == LTC_MP_LT) { + if ((err = mp_add(R->y, modulus, R->y)) != CRYPT_OK) { + goto done; + } + } + /* Y = Y * T1 */ + if ((err = mp_mul(R->y, t1, R->y)) != CRYPT_OK) { + goto done; + } + if ((err = mp_montgomery_reduce(R->y, modulus, mp)) != CRYPT_OK) { + goto done; + } + /* Y = Y - T2 */ + if ((err = mp_sub(R->y, t2, R->y)) != CRYPT_OK) { + goto done; + } + if (mp_cmp_d(R->y, 0) == LTC_MP_LT) { + if ((err = mp_add(R->y, modulus, R->y)) != CRYPT_OK) { + goto done; + } + } + + err = CRYPT_OK; +done: + mp_clear_multi(t1, t2, NULL); + return err; +} +#endif +/* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ltc_ecc_projective_dbl_point.c,v $ */ +/* $Revision: 1.11 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +#define DESC_DEF_ONLY + +#ifdef LTM_DESC + +#undef mp_init +#undef mp_init_multi +#undef mp_clear +#undef mp_clear_multi +#undef mp_init_copy +#undef mp_neg +#undef mp_copy +#undef mp_set +#undef mp_set_int +#undef mp_get_int +#undef mp_get_digit +#undef mp_get_digit_count +#undef mp_cmp +#undef mp_cmp_d +#undef mp_count_bits +#undef mp_cnt_lsb +#undef mp_2expt +#undef mp_read_radix +#undef mp_toradix +#undef mp_unsigned_bin_size +#undef mp_to_unsigned_bin +#undef mp_read_unsigned_bin +#undef mp_add +#undef mp_add_d +#undef mp_sub +#undef mp_sub_d +#undef mp_mul +#undef mp_mul_d +#undef mp_sqr +#undef mp_div +#undef mp_div_2 +#undef mp_mod +#undef mp_mod_d +#undef mp_gcd +#undef mp_lcm +#undef mp_mulmod +#undef mp_sqrmod +#undef mp_invmod +#undef mp_montgomery_setup +#undef mp_montgomery_normalization +#undef mp_montgomery_reduce +#undef mp_montgomery_free +#undef mp_exptmod +#undef mp_prime_is_prime +#undef mp_iszero +#undef mp_isodd +#undef mp_exch +#undef mp_tohex + +static const struct { + int mpi_code, ltc_code; +} mpi_to_ltc_codes[] = { + { MP_OKAY, CRYPT_OK }, + { MP_MEM, CRYPT_MEM }, + { MP_VAL, CRYPT_INVALID_ARG }, +}; + +/** + Convert a MPI error to a LTC error (Possibly the most powerful function ever! Oh wait... no) + @param err The error to convert + @return The equivalent LTC error code or CRYPT_ERROR if none found + */ +static int mpi_to_ltc_error(int err) { + int x; + + for (x = 0; x < (int)(sizeof(mpi_to_ltc_codes) / sizeof(mpi_to_ltc_codes[0])); x++) { + if (err == mpi_to_ltc_codes[x].mpi_code) { + return mpi_to_ltc_codes[x].ltc_code; + } + } + return CRYPT_ERROR; +} + +static int init(void **a) { + int err; + + LTC_ARGCHK(a != NULL); + + *a = XCALLOC(1, sizeof(mp_int)); + if (*a == NULL) { + return CRYPT_MEM; + } + if ((err = mpi_to_ltc_error(mp_init(*a))) != CRYPT_OK) { + XFREE(*a); + } + return err; +} + +static void deinit(void *a) { + LTC_ARGCHKVD(a != NULL); + mp_clear(a); + XFREE(a); +} + +static int neg(void *a, void *b) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_neg(a, b)); +} + +static int copy(void *a, void *b) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_copy(a, b)); +} + +static int init_copy(void **a, void *b) { + if (init(a) != CRYPT_OK) { + return CRYPT_MEM; + } + return copy(b, *a); +} + +/* ---- trivial ---- */ +static int set_int(void *a, unsigned long b) { + LTC_ARGCHK(a != NULL); + return mpi_to_ltc_error(mp_set_int(a, b)); +} + +static unsigned long get_int(void *a) { + LTC_ARGCHK(a != NULL); + return mp_get_int(a); +} + +static unsigned long get_digit(void *a, int n) { + mp_int *A; + + LTC_ARGCHK(a != NULL); + A = a; + return (n >= A->used || n < 0) ? 0 : A->dp[n]; +} + +static int get_digit_count(void *a) { + mp_int *A; + + LTC_ARGCHK(a != NULL); + A = a; + return A->used; +} + +static int compare(void *a, void *b) { + int ret; + + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + ret = mp_cmp(a, b); + switch (ret) { + case MP_LT: + return LTC_MP_LT; + + case MP_EQ: + return LTC_MP_EQ; + + case MP_GT: + return LTC_MP_GT; + } + return 0; +} + +static int compare_d(void *a, unsigned long b) { + int ret; + + LTC_ARGCHK(a != NULL); + ret = mp_cmp_d(a, b); + switch (ret) { + case MP_LT: + return LTC_MP_LT; + + case MP_EQ: + return LTC_MP_EQ; + + case MP_GT: + return LTC_MP_GT; + } + return 0; +} + +static int count_bits(void *a) { + LTC_ARGCHK(a != NULL); + return mp_count_bits(a); +} + +static int count_lsb_bits(void *a) { + LTC_ARGCHK(a != NULL); + return mp_cnt_lsb(a); +} + +static int twoexpt(void *a, int n) { + LTC_ARGCHK(a != NULL); + return mpi_to_ltc_error(mp_2expt(a, n)); +} + +/* ---- conversions ---- */ + +/* read ascii string */ +static int read_radix(void *a, const char *b, int radix) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_read_radix(a, b, radix)); +} + +/* write one */ +static int write_radix(void *a, char *b, int radix) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_toradix(a, b, radix)); +} + +/* get size as unsigned char string */ +static unsigned long unsigned_size(void *a) { + LTC_ARGCHK(a != NULL); + return mp_unsigned_bin_size(a); +} + +/* store */ +static int unsigned_write(void *a, unsigned char *b) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_to_unsigned_bin(a, b)); +} + +/* read */ +static int unsigned_read(void *a, unsigned char *b, unsigned long len) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_read_unsigned_bin(a, b, len)); +} + +/* add */ +static int add(void *a, void *b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_add(a, b, c)); +} + +static int addi(void *a, unsigned long b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_add_d(a, b, c)); +} + +/* sub */ +static int sub(void *a, void *b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_sub(a, b, c)); +} + +static int subi(void *a, unsigned long b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_sub_d(a, b, c)); +} + +/* mul */ +static int mul(void *a, void *b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_mul(a, b, c)); +} + +static int muli(void *a, unsigned long b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_mul_d(a, b, c)); +} + +/* sqr */ +static int sqr(void *a, void *b) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_sqr(a, b)); +} + +/* div */ +static int divide(void *a, void *b, void *c, void *d) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_div(a, b, c, d)); +} + +static int div_2(void *a, void *b) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_div_2(a, b)); +} + +/* modi */ +static int modi(void *a, unsigned long b, unsigned long *c) { + mp_digit tmp; + int err; + + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(c != NULL); + + if ((err = mpi_to_ltc_error(mp_mod_d(a, b, &tmp))) != CRYPT_OK) { + return err; + } + *c = tmp; + return CRYPT_OK; +} + +/* gcd */ +static int gcd(void *a, void *b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_gcd(a, b, c)); +} + +/* lcm */ +static int lcm(void *a, void *b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_lcm(a, b, c)); +} + +static int mulmod(void *a, void *b, void *c, void *d) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + LTC_ARGCHK(d != NULL); + return mpi_to_ltc_error(mp_mulmod(a, b, c, d)); +} + +static int sqrmod(void *a, void *b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_sqrmod(a, b, c)); +} + +/* invmod */ +static int invmod(void *a, void *b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_invmod(a, b, c)); +} + +/* setup */ +static int montgomery_setup(void *a, void **b) { + int err; + + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + *b = XCALLOC(1, sizeof(mp_digit)); + if (*b == NULL) { + return CRYPT_MEM; + } + if ((err = mpi_to_ltc_error(mp_montgomery_setup(a, (mp_digit *)*b))) != CRYPT_OK) { + XFREE(*b); + } + return err; +} + +/* get normalization value */ +static int montgomery_normalization(void *a, void *b) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + return mpi_to_ltc_error(mp_montgomery_calc_normalization(a, b)); +} + +/* reduce */ +static int montgomery_reduce(void *a, void *b, void *c) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + return mpi_to_ltc_error(mp_montgomery_reduce(a, b, *((mp_digit *)c))); +} + +/* clean up */ +static void montgomery_deinit(void *a) { + XFREE(a); +} + +static int exptmod(void *a, void *b, void *c, void *d) { + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + LTC_ARGCHK(c != NULL); + LTC_ARGCHK(d != NULL); + return mpi_to_ltc_error(mp_exptmod(a, b, c, d)); +} + +static int isprime(void *a, int *b) { + int err; + + LTC_ARGCHK(a != NULL); + LTC_ARGCHK(b != NULL); + err = mpi_to_ltc_error(mp_prime_is_prime(a, 8, b)); + *b = (*b == MP_YES) ? LTC_MP_YES : LTC_MP_NO; + return err; +} + +const ltc_math_descriptor ltm_desc = { + "LibTomMath", + (int)DIGIT_BIT, + + &init, + &init_copy, + &deinit, + + &neg, + ©, + + &set_int, + &get_int, + &get_digit, + &get_digit_count, + &compare, + &compare_d, + &count_bits, + &count_lsb_bits, + &twoexpt, + + &read_radix, + &write_radix, + &unsigned_size, + &unsigned_write, + &unsigned_read, + + &add, + &addi, + &sub, + &subi, + &mul, + &muli, + &sqr, + ÷, + &div_2, + &modi, + &gcd, + &lcm, + + &mulmod, + &sqrmod, + &invmod, + + &montgomery_setup, + &montgomery_normalization, + &montgomery_reduce, + &montgomery_deinit, + + &exptmod, + &isprime, + + #ifdef LTC_MECC + #ifdef LTC_MECC_FP + <c_ecc_fp_mulmod, + #else + <c_ecc_mulmod, + #endif + <c_ecc_projective_add_point, + <c_ecc_projective_dbl_point, + <c_ecc_map, + #ifdef LTC_ECC_SHAMIR + #ifdef LTC_MECC_FP + <c_ecc_fp_mul2add, + #else + <c_ecc_mul2add, + #endif /* LTC_MECC_FP */ + #else + NULL, + #endif /* LTC_ECC_SHAMIR */ + #else + NULL, NULL,NULL, NULL, NULL, + #endif /* LTC_MECC */ + + #ifdef LTC_MRSA + &rsa_make_key, + &rsa_exptmod, + #else + NULL, NULL + #endif +}; + + #define mp_init(a) ltc_mp.init(a) + #define mp_init_multi ltc_init_multi + #define mp_clear(a) ltc_mp.deinit(a) + #define mp_clear_multi ltc_deinit_multi + #define mp_init_copy(a, b) ltc_mp.init_copy(a, b) + + #define mp_neg(a, b) ltc_mp.neg(a, b) + #define mp_copy(a, b) ltc_mp.copy(a, b) + + #define mp_set(a, b) ltc_mp.set_int(a, b) + #define mp_set_int(a, b) ltc_mp.set_int(a, b) + #define mp_get_int(a) ltc_mp.get_int(a) + #define mp_get_digit(a, n) ltc_mp.get_digit(a, n) + #define mp_get_digit_count(a) ltc_mp.get_digit_count(a) + #define mp_cmp(a, b) ltc_mp.compare(a, b) + #define mp_cmp_d(a, b) ltc_mp.compare_d(a, b) + #define mp_count_bits(a) ltc_mp.count_bits(a) + #define mp_cnt_lsb(a) ltc_mp.count_lsb_bits(a) + #define mp_2expt(a, b) ltc_mp.twoexpt(a, b) + + #define mp_read_radix(a, b, c) ltc_mp.read_radix(a, b, c) + #define mp_toradix(a, b, c) ltc_mp.write_radix(a, b, c) + #define mp_unsigned_bin_size(a) ltc_mp.unsigned_size(a) + #define mp_to_unsigned_bin(a, b) ltc_mp.unsigned_write(a, b) + #define mp_read_unsigned_bin(a, b, c) ltc_mp.unsigned_read(a, b, c) + + #define mp_add(a, b, c) ltc_mp.add(a, b, c) + #define mp_add_d(a, b, c) ltc_mp.addi(a, b, c) + #define mp_sub(a, b, c) ltc_mp.sub(a, b, c) + #define mp_sub_d(a, b, c) ltc_mp.subi(a, b, c) + #define mp_mul(a, b, c) ltc_mp.mul(a, b, c) + #define mp_mul_d(a, b, c) ltc_mp.muli(a, b, c) + #define mp_sqr(a, b) ltc_mp.sqr(a, b) + #define mp_div(a, b, c, d) ltc_mp.mpdiv(a, b, c, d) + #define mp_div_2(a, b) ltc_mp.div_2(a, b) + #define mp_mod(a, b, c) ltc_mp.mpdiv(a, b, NULL, c) + #define mp_mod_d(a, b, c) ltc_mp.modi(a, b, c) + #define mp_gcd(a, b, c) ltc_mp.gcd(a, b, c) + #define mp_lcm(a, b, c) ltc_mp.lcm(a, b, c) + + #define mp_mulmod(a, b, c, d) ltc_mp.mulmod(a, b, c, d) + #define mp_sqrmod(a, b, c) ltc_mp.sqrmod(a, b, c) + #define mp_invmod(a, b, c) ltc_mp.invmod(a, b, c) + + #define mp_montgomery_setup(a, b) ltc_mp.montgomery_setup(a, b) + #define mp_montgomery_normalization(a, b) ltc_mp.montgomery_normalization(a, b) + #define mp_montgomery_reduce(a, b, c) ltc_mp.montgomery_reduce(a, b, c) + #define mp_montgomery_free(a) ltc_mp.montgomery_deinit(a) + + #define mp_exptmod(a, b, c, d) ltc_mp.exptmod(a, b, c, d) + #define mp_prime_is_prime(a, b, c) ltc_mp.isprime(a, c) + + #define mp_iszero(a) (mp_cmp_d(a, 0) == LTC_MP_EQ ? LTC_MP_YES : LTC_MP_NO) + #define mp_isodd(a) (mp_get_digit_count(a) > 0 ? (mp_get_digit(a, 0) & 1 ? LTC_MP_YES : LTC_MP_NO) : LTC_MP_NO) + #define mp_exch(a, b) do { void *ABC__tmp = a; a = b; b = ABC__tmp; } while (0); + + #define mp_tohex(a, b) mp_toradix(a, b, 16) + +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/math/ltm_desc.c,v $ */ +/* $Revision: 1.31 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +#ifdef MPI +#include + +int ltc_init_multi(void **a, ...) { + void **cur = a; + int np = 0; + va_list args; + + va_start(args, a); + while (cur != NULL) { + if (mp_init(cur) != CRYPT_OK) { + /* failed */ + va_list clean_list; + + va_start(clean_list, a); + cur = a; + while (np--) { + mp_clear(*cur); + cur = va_arg(clean_list, void **); + } + va_end(clean_list); + return CRYPT_MEM; + } + ++np; + cur = va_arg(args, void **); + } + va_end(args); + return CRYPT_OK; +} + +void ltc_deinit_multi(void *a, ...) { + void *cur = a; + va_list args; + + va_start(args, a); + while (cur != NULL) { + mp_clear(cur); + cur = va_arg(args, void *); + } + va_end(args); +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/math/multi.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:23 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file pkcs_1_i2osp.c + Integer to Octet I2OSP, Tom St Denis + */ + +#ifdef LTC_PKCS_1 + +/* always stores the same # of bytes, pads with leading zero bytes + as required + */ + +/** + LTC_PKCS #1 Integer to binary + @param n The integer to store + @param modulus_len The length of the RSA modulus + @param out [out] The destination for the integer + @return CRYPT_OK if successful + */ +int pkcs_1_i2osp(void *n, unsigned long modulus_len, unsigned char *out) { + unsigned long size; + + size = mp_unsigned_bin_size(n); + + if (size > modulus_len) { + return CRYPT_BUFFER_OVERFLOW; + } + + /* store it */ + zeromem(out, modulus_len); + return mp_to_unsigned_bin(n, out + (modulus_len - size)); +} +#endif /* LTC_PKCS_1 */ + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_i2osp.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file pkcs_1_mgf1.c + The Mask Generation Function (MGF1) for LTC_PKCS #1, Tom St Denis + */ + +#ifdef LTC_PKCS_1 + +/** + Perform LTC_PKCS #1 MGF1 (internal) + @param seed The seed for MGF1 + @param seedlen The length of the seed + @param hash_idx The index of the hash desired + @param mask [out] The destination + @param masklen The length of the mask desired + @return CRYPT_OK if successful + */ +int pkcs_1_mgf1(int hash_idx, + const unsigned char *seed, unsigned long seedlen, + unsigned char *mask, unsigned long masklen) { + unsigned long hLen, x; + ulong32 counter; + int err; + hash_state *md; + unsigned char *buf; + + LTC_ARGCHK(seed != NULL); + LTC_ARGCHK(mask != NULL); + + /* ensure valid hash */ + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + + /* get hash output size */ + hLen = hash_descriptor[hash_idx].hashsize; + + /* allocate memory */ + md = XMALLOC(sizeof(hash_state)); + buf = XMALLOC(hLen); + if ((md == NULL) || (buf == NULL)) { + if (md != NULL) { + XFREE(md); + } + if (buf != NULL) { + XFREE(buf); + } + return CRYPT_MEM; + } + + /* start counter */ + counter = 0; + + while (masklen > 0) { + /* handle counter */ + STORE32H(counter, buf); + ++counter; + + /* get hash of seed || counter */ + if ((err = hash_descriptor[hash_idx].init(md)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].process(md, seed, seedlen)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].process(md, buf, 4)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].done(md, buf)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* store it */ + for (x = 0; x < hLen && masklen > 0; x++, masklen--) { + *mask++ = buf[x]; + } + } + + err = CRYPT_OK; +LBL_ERR: + #ifdef LTC_CLEAN_STACK + zeromem(buf, hLen); + zeromem(md, sizeof(hash_state)); + #endif + + XFREE(buf); + XFREE(md); + + return err; +} +#endif /* LTC_PKCS_1 */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_mgf1.c,v $ */ +/* $Revision: 1.8 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file pkcs_1_oaep_decode.c + OAEP Padding for LTC_PKCS #1, Tom St Denis + */ + +#ifdef LTC_PKCS_1 + +/** + LTC_PKCS #1 v2.00 OAEP decode + @param msg The encoded data to decode + @param msglen The length of the encoded data (octets) + @param lparam The session or system data (can be NULL) + @param lparamlen The length of the lparam + @param modulus_bitlen The bit length of the RSA modulus + @param hash_idx The index of the hash desired + @param out [out] Destination of decoding + @param outlen [in/out] The max size and resulting size of the decoding + @param res [out] Result of decoding, 1==valid, 0==invalid + @return CRYPT_OK if successful (even if invalid) + */ +int pkcs_1_oaep_decode(const unsigned char *msg, unsigned long msglen, + const unsigned char *lparam, unsigned long lparamlen, + unsigned long modulus_bitlen, int hash_idx, + unsigned char *out, unsigned long *outlen, + int *res) { + unsigned char *DB, *seed, *mask; + unsigned long hLen, x, y, modulus_len; + int err; + + LTC_ARGCHK(msg != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(res != NULL); + + /* default to invalid packet */ + *res = 0; + + /* test valid hash */ + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + hLen = hash_descriptor[hash_idx].hashsize; + modulus_len = (modulus_bitlen >> 3) + (modulus_bitlen & 7 ? 1 : 0); + + /* test hash/message size */ + if ((2 * hLen >= (modulus_len - 2)) || (msglen != modulus_len)) { + return CRYPT_PK_INVALID_SIZE; + } + + /* allocate ram for DB/mask/salt of size modulus_len */ + DB = XMALLOC(modulus_len); + mask = XMALLOC(modulus_len); + seed = XMALLOC(hLen); + if ((DB == NULL) || (mask == NULL) || (seed == NULL)) { + if (DB != NULL) { + XFREE(DB); + } + if (mask != NULL) { + XFREE(mask); + } + if (seed != NULL) { + XFREE(seed); + } + return CRYPT_MEM; + } + + /* ok so it's now in the form + + 0x00 || maskedseed || maskedDB + + 1 || hLen || modulus_len - hLen - 1 + + */ + + /* must have leading 0x00 byte */ + if (msg[0] != 0x00) { + err = CRYPT_OK; + goto LBL_ERR; + } + + /* now read the masked seed */ + x = 1; + XMEMCPY(seed, msg + x, hLen); + x += hLen; + + /* now read the masked DB */ + XMEMCPY(DB, msg + x, modulus_len - hLen - 1); + x += modulus_len - hLen - 1; + + /* compute MGF1 of maskedDB (hLen) */ + if ((err = pkcs_1_mgf1(hash_idx, DB, modulus_len - hLen - 1, mask, hLen)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* XOR against seed */ + for (y = 0; y < hLen; y++) { + seed[y] ^= mask[y]; + } + + /* compute MGF1 of seed (k - hlen - 1) */ + if ((err = pkcs_1_mgf1(hash_idx, seed, hLen, mask, modulus_len - hLen - 1)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* xor against DB */ + for (y = 0; y < (modulus_len - hLen - 1); y++) { + DB[y] ^= mask[y]; + } + + /* now DB == lhash || PS || 0x01 || M, PS == k - mlen - 2hlen - 2 zeroes */ + + /* compute lhash and store it in seed [reuse temps!] */ + x = modulus_len; + if (lparam != NULL) { + if ((err = hash_memory(hash_idx, lparam, lparamlen, seed, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + } else { + /* can't pass hash_memory a NULL so use DB with zero length */ + if ((err = hash_memory(hash_idx, DB, 0, seed, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + } + + /* compare the lhash'es */ + if (XMEMCMP(seed, DB, hLen) != 0) { + err = CRYPT_OK; + goto LBL_ERR; + } + + /* now zeroes before a 0x01 */ + for (x = hLen; x < (modulus_len - hLen - 1) && DB[x] == 0x00; x++) { + /* step... */ + } + + /* error out if wasn't 0x01 */ + if ((x == (modulus_len - hLen - 1)) || (DB[x] != 0x01)) { + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + + /* rest is the message (and skip 0x01) */ + if ((modulus_len - hLen - 1 - ++x) > *outlen) { + *outlen = modulus_len - hLen - 1 - x; + err = CRYPT_BUFFER_OVERFLOW; + goto LBL_ERR; + } + + /* copy message */ + *outlen = modulus_len - hLen - 1 - x; + XMEMCPY(out, DB + x, modulus_len - hLen - 1 - x); + x += modulus_len - hLen - 1; + + /* valid packet */ + *res = 1; + + err = CRYPT_OK; +LBL_ERR: + #ifdef LTC_CLEAN_STACK + zeromem(DB, modulus_len); + zeromem(seed, hLen); + zeromem(mask, modulus_len); + #endif + + XFREE(seed); + XFREE(mask); + XFREE(DB); + + return err; +} +#endif /* LTC_PKCS_1 */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_oaep_decode.c,v $ */ +/* $Revision: 1.13 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file pkcs_1_oaep_encode.c + OAEP Padding for LTC_PKCS #1, Tom St Denis + */ + +#ifdef LTC_PKCS_1 + +/** + LTC_PKCS #1 v2.00 OAEP encode + @param msg The data to encode + @param msglen The length of the data to encode (octets) + @param lparam A session or system parameter (can be NULL) + @param lparamlen The length of the lparam data + @param modulus_bitlen The bit length of the RSA modulus + @param prng An active PRNG state + @param prng_idx The index of the PRNG desired + @param hash_idx The index of the hash desired + @param out [out] The destination for the encoded data + @param outlen [in/out] The max size and resulting size of the encoded data + @return CRYPT_OK if successful + */ +int pkcs_1_oaep_encode(const unsigned char *msg, unsigned long msglen, + const unsigned char *lparam, unsigned long lparamlen, + unsigned long modulus_bitlen, prng_state *prng, + int prng_idx, int hash_idx, + unsigned char *out, unsigned long *outlen) { + unsigned char *DB, *seed, *mask; + unsigned long hLen, x, y, modulus_len; + int err; + + LTC_ARGCHK(msg != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* test valid hash */ + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + + /* valid prng */ + if ((err = prng_is_valid(prng_idx)) != CRYPT_OK) { + return err; + } + + hLen = hash_descriptor[hash_idx].hashsize; + modulus_len = (modulus_bitlen >> 3) + (modulus_bitlen & 7 ? 1 : 0); + + /* test message size */ + if ((2 * hLen >= (modulus_len - 2)) || (msglen > (modulus_len - 2 * hLen - 2))) { + return CRYPT_PK_INVALID_SIZE; + } + + /* allocate ram for DB/mask/salt of size modulus_len */ + DB = XMALLOC(modulus_len); + mask = XMALLOC(modulus_len); + seed = XMALLOC(hLen); + if ((DB == NULL) || (mask == NULL) || (seed == NULL)) { + if (DB != NULL) { + XFREE(DB); + } + if (mask != NULL) { + XFREE(mask); + } + if (seed != NULL) { + XFREE(seed); + } + return CRYPT_MEM; + } + + /* get lhash */ + /* DB == lhash || PS || 0x01 || M, PS == k - mlen - 2hlen - 2 zeroes */ + x = modulus_len; + if (lparam != NULL) { + if ((err = hash_memory(hash_idx, lparam, lparamlen, DB, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + } else { + /* can't pass hash_memory a NULL so use DB with zero length */ + if ((err = hash_memory(hash_idx, DB, 0, DB, &x)) != CRYPT_OK) { + goto LBL_ERR; + } + } + + /* append PS then 0x01 (to lhash) */ + x = hLen; + y = modulus_len - msglen - 2 * hLen - 2; + XMEMSET(DB + x, 0, y); + x += y; + + /* 0x01 byte */ + DB[x++] = 0x01; + + /* message (length = msglen) */ + XMEMCPY(DB + x, msg, msglen); + x += msglen; + + /* now choose a random seed */ + if (prng_descriptor[prng_idx].read(seed, hLen, prng) != hLen) { + err = CRYPT_ERROR_READPRNG; + goto LBL_ERR; + } + + /* compute MGF1 of seed (k - hlen - 1) */ + if ((err = pkcs_1_mgf1(hash_idx, seed, hLen, mask, modulus_len - hLen - 1)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* xor against DB */ + for (y = 0; y < (modulus_len - hLen - 1); y++) { + DB[y] ^= mask[y]; + } + + /* compute MGF1 of maskedDB (hLen) */ + if ((err = pkcs_1_mgf1(hash_idx, DB, modulus_len - hLen - 1, mask, hLen)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* XOR against seed */ + for (y = 0; y < hLen; y++) { + seed[y] ^= mask[y]; + } + + /* create string of length modulus_len */ + if (*outlen < modulus_len) { + *outlen = modulus_len; + err = CRYPT_BUFFER_OVERFLOW; + goto LBL_ERR; + } + + /* start output which is 0x00 || maskedSeed || maskedDB */ + x = 0; + out[x++] = 0x00; + XMEMCPY(out + x, seed, hLen); + x += hLen; + XMEMCPY(out + x, DB, modulus_len - hLen - 1); + x += modulus_len - hLen - 1; + + *outlen = x; + + err = CRYPT_OK; +LBL_ERR: + #ifdef LTC_CLEAN_STACK + zeromem(DB, modulus_len); + zeromem(seed, hLen); + zeromem(mask, modulus_len); + #endif + + XFREE(seed); + XFREE(mask); + XFREE(DB); + + return err; +} +#endif /* LTC_PKCS_1 */ + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_oaep_encode.c,v $ */ +/* $Revision: 1.9 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file pkcs_1_os2ip.c + Octet to Integer OS2IP, Tom St Denis + */ +#ifdef LTC_PKCS_1 + +/** + Read a binary string into an mp_int + @param n [out] The mp_int destination + @param in The binary string to read + @param inlen The length of the binary string + @return CRYPT_OK if successful + */ +int pkcs_1_os2ip(void *n, unsigned char *in, unsigned long inlen) { + return mp_read_unsigned_bin(n, in, inlen); +} +#endif /* LTC_PKCS_1 */ + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_os2ip.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file pkcs_1_pss_decode.c + LTC_PKCS #1 PSS Signature Padding, Tom St Denis + */ + +#ifdef LTC_PKCS_1 + +/** + LTC_PKCS #1 v2.00 PSS decode + @param msghash The hash to verify + @param msghashlen The length of the hash (octets) + @param sig The signature data (encoded data) + @param siglen The length of the signature data (octets) + @param saltlen The length of the salt used (octets) + @param hash_idx The index of the hash desired + @param modulus_bitlen The bit length of the RSA modulus + @param res [out] The result of the comparison, 1==valid, 0==invalid + @return CRYPT_OK if successful (even if the comparison failed) + */ +int pkcs_1_pss_decode(const unsigned char *msghash, unsigned long msghashlen, + const unsigned char *sig, unsigned long siglen, + unsigned long saltlen, int hash_idx, + unsigned long modulus_bitlen, int *res) { + unsigned char *DB, *mask, *salt, *hash; + unsigned long x, y, hLen, modulus_len; + int err; + hash_state md; + + LTC_ARGCHK(msghash != NULL); + LTC_ARGCHK(res != NULL); + + /* default to invalid */ + *res = 0; + + /* ensure hash is valid */ + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + + hLen = hash_descriptor[hash_idx].hashsize; + modulus_len = (modulus_bitlen >> 3) + (modulus_bitlen & 7 ? 1 : 0); + + /* check sizes */ + if ((saltlen > modulus_len) || + (modulus_len < hLen + saltlen + 2) || (siglen != modulus_len)) { + return CRYPT_PK_INVALID_SIZE; + } + + /* allocate ram for DB/mask/salt/hash of size modulus_len */ + DB = XMALLOC(modulus_len); + mask = XMALLOC(modulus_len); + salt = XMALLOC(modulus_len); + hash = XMALLOC(modulus_len); + if ((DB == NULL) || (mask == NULL) || (salt == NULL) || (hash == NULL)) { + if (DB != NULL) { + XFREE(DB); + } + if (mask != NULL) { + XFREE(mask); + } + if (salt != NULL) { + XFREE(salt); + } + if (hash != NULL) { + XFREE(hash); + } + return CRYPT_MEM; + } + + /* ensure the 0xBC byte */ + if (sig[siglen - 1] != 0xBC) { + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + + /* copy out the DB */ + x = 0; + XMEMCPY(DB, sig + x, modulus_len - hLen - 1); + x += modulus_len - hLen - 1; + + /* copy out the hash */ + XMEMCPY(hash, sig + x, hLen); + x += hLen; + + /* check the MSB */ + if ((sig[0] & ~(0xFF >> ((modulus_len << 3) - (modulus_bitlen - 1)))) != 0) { + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + + /* generate mask of length modulus_len - hLen - 1 from hash */ + if ((err = pkcs_1_mgf1(hash_idx, hash, hLen, mask, modulus_len - hLen - 1)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* xor against DB */ + for (y = 0; y < (modulus_len - hLen - 1); y++) { + DB[y] ^= mask[y]; + } + + /* now clear the first byte [make sure smaller than modulus] */ + DB[0] &= 0xFF >> ((modulus_len << 3) - (modulus_bitlen - 1)); + + /* DB = PS || 0x01 || salt, PS == modulus_len - saltlen - hLen - 2 zero bytes */ + + /* check for zeroes and 0x01 */ + for (x = 0; x < modulus_len - saltlen - hLen - 2; x++) { + if (DB[x] != 0x00) { + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + } + + /* check for the 0x01 */ + if (DB[x++] != 0x01) { + err = CRYPT_INVALID_PACKET; + goto LBL_ERR; + } + + /* M = (eight) 0x00 || msghash || salt, mask = H(M) */ + if ((err = hash_descriptor[hash_idx].init(&md)) != CRYPT_OK) { + goto LBL_ERR; + } + zeromem(mask, 8); + if ((err = hash_descriptor[hash_idx].process(&md, mask, 8)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].process(&md, msghash, msghashlen)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].process(&md, DB + x, saltlen)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].done(&md, mask)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* mask == hash means valid signature */ + if (XMEMCMP(mask, hash, hLen) == 0) { + *res = 1; + } + + err = CRYPT_OK; +LBL_ERR: + #ifdef LTC_CLEAN_STACK + zeromem(DB, modulus_len); + zeromem(mask, modulus_len); + zeromem(salt, modulus_len); + zeromem(hash, modulus_len); + #endif + + XFREE(hash); + XFREE(salt); + XFREE(mask); + XFREE(DB); + + return err; +} +#endif /* LTC_PKCS_1 */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_pss_decode.c,v $ */ +/* $Revision: 1.11 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file pkcs_1_pss_encode.c + LTC_PKCS #1 PSS Signature Padding, Tom St Denis + */ + +#ifdef LTC_PKCS_1 + +/** + LTC_PKCS #1 v2.00 Signature Encoding + @param msghash The hash to encode + @param msghashlen The length of the hash (octets) + @param saltlen The length of the salt desired (octets) + @param prng An active PRNG context + @param prng_idx The index of the PRNG desired + @param hash_idx The index of the hash desired + @param modulus_bitlen The bit length of the RSA modulus + @param out [out] The destination of the encoding + @param outlen [in/out] The max size and resulting size of the encoded data + @return CRYPT_OK if successful + */ +int pkcs_1_pss_encode(const unsigned char *msghash, unsigned long msghashlen, + unsigned long saltlen, prng_state *prng, + int prng_idx, int hash_idx, + unsigned long modulus_bitlen, + unsigned char *out, unsigned long *outlen) { + unsigned char *DB, *mask, *salt, *hash; + unsigned long x, y, hLen, modulus_len; + int err; + hash_state md; + + LTC_ARGCHK(msghash != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + + /* ensure hash and PRNG are valid */ + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + if ((err = prng_is_valid(prng_idx)) != CRYPT_OK) { + return err; + } + + hLen = hash_descriptor[hash_idx].hashsize; + modulus_len = (modulus_bitlen >> 3) + (modulus_bitlen & 7 ? 1 : 0); + + /* check sizes */ + if ((saltlen > modulus_len) || (modulus_len < hLen + saltlen + 2)) { + return CRYPT_PK_INVALID_SIZE; + } + + /* allocate ram for DB/mask/salt/hash of size modulus_len */ + DB = XMALLOC(modulus_len); + mask = XMALLOC(modulus_len); + salt = XMALLOC(modulus_len); + hash = XMALLOC(modulus_len); + if ((DB == NULL) || (mask == NULL) || (salt == NULL) || (hash == NULL)) { + if (DB != NULL) { + XFREE(DB); + } + if (mask != NULL) { + XFREE(mask); + } + if (salt != NULL) { + XFREE(salt); + } + if (hash != NULL) { + XFREE(hash); + } + return CRYPT_MEM; + } + + + /* generate random salt */ + if (saltlen > 0) { + if (prng_descriptor[prng_idx].read(salt, saltlen, prng) != saltlen) { + err = CRYPT_ERROR_READPRNG; + goto LBL_ERR; + } + } + + /* M = (eight) 0x00 || msghash || salt, hash = H(M) */ + if ((err = hash_descriptor[hash_idx].init(&md)) != CRYPT_OK) { + goto LBL_ERR; + } + zeromem(DB, 8); + if ((err = hash_descriptor[hash_idx].process(&md, DB, 8)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].process(&md, msghash, msghashlen)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].process(&md, salt, saltlen)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash_idx].done(&md, hash)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* generate DB = PS || 0x01 || salt, PS == modulus_len - saltlen - hLen - 2 zero bytes */ + x = 0; + XMEMSET(DB + x, 0, modulus_len - saltlen - hLen - 2); + x += modulus_len - saltlen - hLen - 2; + DB[x++] = 0x01; + XMEMCPY(DB + x, salt, saltlen); + x += saltlen; + + /* generate mask of length modulus_len - hLen - 1 from hash */ + if ((err = pkcs_1_mgf1(hash_idx, hash, hLen, mask, modulus_len - hLen - 1)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* xor against DB */ + for (y = 0; y < (modulus_len - hLen - 1); y++) { + DB[y] ^= mask[y]; + } + + /* output is DB || hash || 0xBC */ + if (*outlen < modulus_len) { + *outlen = modulus_len; + err = CRYPT_BUFFER_OVERFLOW; + goto LBL_ERR; + } + + /* DB len = modulus_len - hLen - 1 */ + y = 0; + XMEMCPY(out + y, DB, modulus_len - hLen - 1); + y += modulus_len - hLen - 1; + + /* hash */ + XMEMCPY(out + y, hash, hLen); + y += hLen; + + /* 0xBC */ + out[y] = 0xBC; + + /* now clear the 8*modulus_len - modulus_bitlen most significant bits */ + out[0] &= 0xFF >> ((modulus_len << 3) - (modulus_bitlen - 1)); + + /* store output size */ + *outlen = modulus_len; + err = CRYPT_OK; +LBL_ERR: + #ifdef LTC_CLEAN_STACK + zeromem(DB, modulus_len); + zeromem(mask, modulus_len); + zeromem(salt, modulus_len); + zeromem(hash, modulus_len); + #endif + + XFREE(hash); + XFREE(salt); + XFREE(mask); + XFREE(DB); + + return err; +} +#endif /* LTC_PKCS_1 */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_pss_encode.c,v $ */ +/* $Revision: 1.9 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** @file pkcs_1_v1_5_decode.c + * + * LTC_PKCS #1 v1.5 Padding. (Andreas Lange) + */ + +#ifdef LTC_PKCS_1 + +/** @brief LTC_PKCS #1 v1.5 decode. + * + * @param msg The encoded data to decode + * @param msglen The length of the encoded data (octets) + * @param block_type Block type to use in padding (\sa ltc_pkcs_1_v1_5_blocks) + * @param modulus_bitlen The bit length of the RSA modulus + * @param out [out] Destination of decoding + * @param outlen [in/out] The max size and resulting size of the decoding + * @param is_valid [out] Boolean whether the padding was valid + * + * @return CRYPT_OK if successful (even if invalid) + */ +int pkcs_1_v1_5_decode(const unsigned char *msg, + unsigned long msglen, + int block_type, + unsigned long modulus_bitlen, + unsigned char *out, + unsigned long *outlen, + int *is_valid) { + unsigned long modulus_len, ps_len, i; + int result; + + /* default to invalid packet */ + *is_valid = 0; + + modulus_len = (modulus_bitlen >> 3) + (modulus_bitlen & 7 ? 1 : 0); + + /* test message size */ + + if ((msglen > modulus_len) || (modulus_len < 11)) { + return CRYPT_PK_INVALID_SIZE; + } + + /* separate encoded message */ + + if ((msg[0] != 0x00) || (msg[1] != (unsigned char)block_type)) { + result = CRYPT_INVALID_PACKET; + goto bail; + } + + if (block_type == LTC_LTC_PKCS_1_EME) { + for (i = 2; i < modulus_len; i++) { + /* separator */ + if (msg[i] == 0x00) { + break; + } + } + ps_len = i++ - 2; + + if ((i >= modulus_len) || (ps_len < 8)) { + /* There was no octet with hexadecimal value 0x00 to separate ps from m, + * or the length of ps is less than 8 octets. + */ + result = CRYPT_INVALID_PACKET; + goto bail; + } + } else { + for (i = 2; i < modulus_len - 1; i++) { + if (msg[i] != 0xFF) { + break; + } + } + + /* separator check */ + if (msg[i] != 0) { + /* There was no octet with hexadecimal value 0x00 to separate ps from m. */ + result = CRYPT_INVALID_PACKET; + goto bail; + } + + ps_len = i - 2; + } + + if (*outlen < (msglen - (2 + ps_len + 1))) { + *outlen = msglen - (2 + ps_len + 1); + result = CRYPT_BUFFER_OVERFLOW; + goto bail; + } + + *outlen = (msglen - (2 + ps_len + 1)); + XMEMCPY(out, &msg[2 + ps_len + 1], *outlen); + + /* valid packet */ + *is_valid = 1; + result = CRYPT_OK; +bail: + return result; +} /* pkcs_1_v1_5_decode */ +#endif /* #ifdef LTC_PKCS_1 */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_v1_5_decode.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/*! \file pkcs_1_v1_5_encode.c + * + * LTC_PKCS #1 v1.5 Padding (Andreas Lange) + */ + +#ifdef LTC_PKCS_1 + +/*! \brief LTC_PKCS #1 v1.5 encode. + * + * \param msg The data to encode + * \param msglen The length of the data to encode (octets) + * \param block_type Block type to use in padding (\sa ltc_pkcs_1_v1_5_blocks) + * \param modulus_bitlen The bit length of the RSA modulus + * \param prng An active PRNG state (only for LTC_LTC_PKCS_1_EME) + * \param prng_idx The index of the PRNG desired (only for LTC_LTC_PKCS_1_EME) + * \param out [out] The destination for the encoded data + * \param outlen [in/out] The max size and resulting size of the encoded data + * + * \return CRYPT_OK if successful + */ +int pkcs_1_v1_5_encode(const unsigned char *msg, + unsigned long msglen, + int block_type, + unsigned long modulus_bitlen, + prng_state *prng, + int prng_idx, + unsigned char *out, + unsigned long *outlen) { + unsigned long modulus_len, ps_len, i; + unsigned char *ps; + int result; + + /* valid block_type? */ + if ((block_type != LTC_LTC_PKCS_1_EMSA) && + (block_type != LTC_LTC_PKCS_1_EME)) { + return CRYPT_PK_INVALID_PADDING; + } + + if (block_type == LTC_LTC_PKCS_1_EME) { /* encryption padding, we need a valid PRNG */ + if ((result = prng_is_valid(prng_idx)) != CRYPT_OK) { + return result; + } + } + + modulus_len = (modulus_bitlen >> 3) + (modulus_bitlen & 7 ? 1 : 0); + + /* test message size */ + if ((msglen + 11) > modulus_len) { + return CRYPT_PK_INVALID_SIZE; + } + + if (*outlen < modulus_len) { + *outlen = modulus_len; + result = CRYPT_BUFFER_OVERFLOW; + goto bail; + } + + /* generate an octets string PS */ + ps = &out[2]; + ps_len = modulus_len - msglen - 3; + + if (block_type == LTC_LTC_PKCS_1_EME) { + /* now choose a random ps */ + if (prng_descriptor[prng_idx].read(ps, ps_len, prng) != ps_len) { + result = CRYPT_ERROR_READPRNG; + goto bail; + } + + /* transform zero bytes (if any) to non-zero random bytes */ + for (i = 0; i < ps_len; i++) { + while (ps[i] == 0) { + if (prng_descriptor[prng_idx].read(&ps[i], 1, prng) != 1) { + result = CRYPT_ERROR_READPRNG; + goto bail; + } + } + } + } else { + XMEMSET(ps, 0xFF, ps_len); + } + + /* create string of length modulus_len */ + out[0] = 0x00; + out[1] = (unsigned char)block_type;/* block_type 1 or 2 */ + out[2 + ps_len] = 0x00; + XMEMCPY(&out[2 + ps_len + 1], msg, msglen); + *outlen = modulus_len; + + result = CRYPT_OK; +bail: + return result; +} /* pkcs_1_v1_5_encode */ +#endif /* #ifdef LTC_PKCS_1 */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/pkcs1/pkcs_1_v1_5_encode.c,v $ */ +/* $Revision: 1.4 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rand_prime.c + Generate a random prime, Tom St Denis + */ + +#define USE_BBS 1 + +int rand_prime(void *N, long len, prng_state *prng, int wprng) { + int err, res, type; + unsigned char *buf; + + LTC_ARGCHK(N != NULL); + + /* get type */ + if (len < 0) { + type = USE_BBS; + len = -len; + } else { + type = 0; + } + + /* allow sizes between 2 and 512 bytes for a prime size */ + if ((len < 2) || (len > 512)) { + return CRYPT_INVALID_PRIME_SIZE; + } + + /* valid PRNG? Better be! */ + if ((err = prng_is_valid(wprng)) != CRYPT_OK) { + return err; + } + + /* allocate buffer to work with */ + buf = XCALLOC(1, len); + if (buf == NULL) { + return CRYPT_MEM; + } + + do { + /* generate value */ + if (prng_descriptor[wprng].read(buf, len, prng) != (unsigned long)len) { + XFREE(buf); + return CRYPT_ERROR_READPRNG; + } + + /* munge bits */ + buf[0] |= 0x80 | 0x40; + buf[len - 1] |= 0x01 | ((type & USE_BBS) ? 0x02 : 0x00); + + /* load value */ + if ((err = mp_read_unsigned_bin(N, buf, len)) != CRYPT_OK) { + XFREE(buf); + return err; + } + + /* test */ + if ((err = mp_prime_is_prime(N, 8, &res)) != CRYPT_OK) { + XFREE(buf); + return err; + } + } while (res == LTC_MP_NO); + +#ifdef LTC_CLEAN_STACK + zeromem(buf, len); +#endif + + XFREE(buf); + return CRYPT_OK; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/math/rand_prime.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:23 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rng_get_bytes.c + portable way to get secure random bits to feed a PRNG (Tom St Denis) + */ + +#ifdef LTC_DEVRANDOM +/* on *NIX read /dev/random */ +static unsigned long rng_nix(unsigned char *buf, unsigned long len, + void (*callback)(void)) { + #ifdef LTC_NO_FILE + return 0; + #else + FILE *f; + unsigned long x; + #ifdef TRY_URANDOM_FIRST + f = fopen("/dev/urandom", "rb"); + if (f == NULL) + #endif /* TRY_URANDOM_FIRST */ + f = fopen("/dev/random", "rb"); + + if (f == NULL) { + return 0; + } + + /* disable buffering */ + if (setvbuf(f, NULL, _IONBF, 0) != 0) { + fclose(f); + return 0; + } + + x = (unsigned long)fread(buf, 1, (size_t)len, f); + fclose(f); + return x; + #endif /* LTC_NO_FILE */ +} +#endif /* LTC_DEVRANDOM */ + +/* on ANSI C platforms with 100 < CLOCKS_PER_SEC < 10000 */ +#if defined(CLOCKS_PER_SEC) && !defined(WINCE) + + #define ANSI_RNG + +static unsigned long rng_ansic(unsigned char *buf, unsigned long len, + void (*callback)(void)) { + clock_t t1; + int l, acc, bits, a, b; + + if ((XCLOCKS_PER_SEC < 100) || (XCLOCKS_PER_SEC > 10000)) { + return 0; + } + + l = len; + bits = 8; + acc = a = b = 0; + while (len--) { + if (callback != NULL) callback(); + while (bits--) { + do { + t1 = XCLOCK(); + while (t1 == XCLOCK()) a ^= 1; + t1 = XCLOCK(); + while (t1 == XCLOCK()) b ^= 1; + } while (a == b); + acc = (acc << 1) | a; + } + *buf++ = acc; + acc = 0; + bits = 8; + } + acc = bits = a = b = 0; + return l; +} +#endif + +/* Try the Microsoft CSP */ +#if defined(WIN32) || defined(WINCE) + #define _WIN32_WINNT 0x0400 + #ifdef WINCE + #define UNDER_CE + #define ARM + #endif +#include +#include + +static unsigned long rng_win32(unsigned char *buf, unsigned long len, + void (*callback)(void)) { + HCRYPTPROV hProv = 0; + + if (!CryptAcquireContext(&hProv, NULL, MS_DEF_PROV, PROV_RSA_FULL, + (CRYPT_VERIFYCONTEXT | CRYPT_MACHINE_KEYSET)) && + !CryptAcquireContext(&hProv, NULL, MS_DEF_PROV, PROV_RSA_FULL, + CRYPT_VERIFYCONTEXT | CRYPT_MACHINE_KEYSET | CRYPT_NEWKEYSET)) + return 0; + + if (CryptGenRandom(hProv, len, buf) == TRUE) { + CryptReleaseContext(hProv, 0); + return len; + } else { + CryptReleaseContext(hProv, 0); + return 0; + } +} +#endif /* WIN32 */ + +/** + Read the system RNG + @param out Destination + @param outlen Length desired (octets) + @param callback Pointer to void function to act as "callback" when RNG is slow. This can be NULL + @return Number of octets read + */ +unsigned long rng_get_bytes(unsigned char *out, unsigned long outlen, + void (*callback)(void)) { + for (unsigned long i = 0; i < outlen; i++) + out[i] = rand() & 0xff; + return outlen; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/prngs/rng_get_bytes.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rng_make_prng.c + portable way to get secure random bits to feed a PRNG (Tom St Denis) + */ + +/** + Create a PRNG from a RNG + @param bits Number of bits of entropy desired (64 ... 1024) + @param wprng Index of which PRNG to setup + @param prng [out] PRNG state to initialize + @param callback A pointer to a void function for when the RNG is slow, this can be NULL + @return CRYPT_OK if successful + */ +int rng_make_prng(int bits, int wprng, prng_state *prng, + void (*callback)(void)) { + unsigned char buf[256]; + int err; + + LTC_ARGCHK(prng != NULL); + + /* check parameter */ + if ((err = prng_is_valid(wprng)) != CRYPT_OK) { + return err; + } + + if ((bits < 64) || (bits > 1024)) { + return CRYPT_INVALID_PRNGSIZE; + } + + if ((err = prng_descriptor[wprng].start(prng)) != CRYPT_OK) { + return err; + } + + bits = ((bits / 8) + ((bits & 7) != 0 ? 1 : 0)) * 2; + if (rng_get_bytes(buf, (unsigned long)bits, callback) != (unsigned long)bits) { + return CRYPT_ERROR_READPRNG; + } + + if ((err = prng_descriptor[wprng].add_entropy(buf, (unsigned long)bits, prng)) != CRYPT_OK) { + return err; + } + + if ((err = prng_descriptor[wprng].ready(prng)) != CRYPT_OK) { + return err; + } + +#ifdef LTC_CLEAN_STACK + zeromem(buf, sizeof(buf)); +#endif + return CRYPT_OK; +} + +/* $Source: /cvs/libtom/libtomcrypt/src/prngs/rng_make_prng.c,v $ */ +/* $Revision: 1.5 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rsa_decrypt_key.c + RSA LTC_PKCS #1 Decryption, Tom St Denis and Andreas Lange + */ + +#ifdef LTC_MRSA + +/** + LTC_PKCS #1 decrypt then v1.5 or OAEP depad + @param in The ciphertext + @param inlen The length of the ciphertext (octets) + @param out [out] The plaintext + @param outlen [in/out] The max size and resulting size of the plaintext (octets) + @param lparam The system "lparam" value + @param lparamlen The length of the lparam value (octets) + @param hash_idx The index of the hash desired + @param padding Type of padding (LTC_LTC_PKCS_1_OAEP or LTC_LTC_PKCS_1_V1_5) + @param stat [out] Result of the decryption, 1==valid, 0==invalid + @param key The corresponding private RSA key + @return CRYPT_OK if succcessul (even if invalid) + */ +int rsa_decrypt_key_ex(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + const unsigned char *lparam, unsigned long lparamlen, + int hash_idx, int padding, + int *stat, rsa_key *key) { + unsigned long modulus_bitlen, modulus_bytelen, x; + int err; + unsigned char *tmp; + + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(key != NULL); + LTC_ARGCHK(stat != NULL); + + /* default to invalid */ + *stat = 0; + + /* valid padding? */ + + if ((padding != LTC_LTC_PKCS_1_V1_5) && + (padding != LTC_LTC_PKCS_1_OAEP)) { + return CRYPT_PK_INVALID_PADDING; + } + + if (padding == LTC_LTC_PKCS_1_OAEP) { + /* valid hash ? */ + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + } + + /* get modulus len in bits */ + modulus_bitlen = mp_count_bits((key->N)); + + /* outlen must be at least the size of the modulus */ + modulus_bytelen = mp_unsigned_bin_size((key->N)); + if (modulus_bytelen != inlen) { + return CRYPT_INVALID_PACKET; + } + + /* allocate ram */ + tmp = XMALLOC(inlen); + if (tmp == NULL) { + return CRYPT_MEM; + } + + /* rsa decode the packet */ + x = inlen; + if ((err = ltc_mp.rsa_me(in, inlen, tmp, &x, PK_PRIVATE, key)) != CRYPT_OK) { + XFREE(tmp); + return err; + } + + if (padding == LTC_LTC_PKCS_1_OAEP) { + /* now OAEP decode the packet */ + err = pkcs_1_oaep_decode(tmp, x, lparam, lparamlen, modulus_bitlen, hash_idx, + out, outlen, stat); + } else { + /* now LTC_PKCS #1 v1.5 depad the packet */ + err = pkcs_1_v1_5_decode(tmp, x, LTC_LTC_PKCS_1_EME, modulus_bitlen, out, outlen, stat); + } + + XFREE(tmp); + return err; +} +#endif /* LTC_MRSA */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/rsa/rsa_decrypt_key.c,v $ */ +/* $Revision: 1.10 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rsa_encrypt_key.c + RSA LTC_PKCS #1 encryption, Tom St Denis and Andreas Lange + */ + +#ifdef LTC_MRSA + +/** + (LTC_PKCS #1 v2.0) OAEP pad then encrypt + @param in The plaintext + @param inlen The length of the plaintext (octets) + @param out [out] The ciphertext + @param outlen [in/out] The max size and resulting size of the ciphertext + @param lparam The system "lparam" for the encryption + @param lparamlen The length of lparam (octets) + @param prng An active PRNG + @param prng_idx The index of the desired prng + @param hash_idx The index of the desired hash + @param padding Type of padding (LTC_LTC_PKCS_1_OAEP or LTC_LTC_PKCS_1_V1_5) + @param key The RSA key to encrypt to + @return CRYPT_OK if successful + */ +int rsa_encrypt_key_ex(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + const unsigned char *lparam, unsigned long lparamlen, + prng_state *prng, int prng_idx, int hash_idx, int padding, rsa_key *key) { + unsigned long modulus_bitlen, modulus_bytelen, x; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(key != NULL); + + /* valid padding? */ + if ((padding != LTC_LTC_PKCS_1_V1_5) && + (padding != LTC_LTC_PKCS_1_OAEP)) { + return CRYPT_PK_INVALID_PADDING; + } + + /* valid prng? */ + if ((err = prng_is_valid(prng_idx)) != CRYPT_OK) { + return err; + } + + if (padding == LTC_LTC_PKCS_1_OAEP) { + /* valid hash? */ + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + } + + /* get modulus len in bits */ + modulus_bitlen = mp_count_bits((key->N)); + + /* outlen must be at least the size of the modulus */ + modulus_bytelen = mp_unsigned_bin_size((key->N)); + if (modulus_bytelen > *outlen) { + *outlen = modulus_bytelen; + return CRYPT_BUFFER_OVERFLOW; + } + + if (padding == LTC_LTC_PKCS_1_OAEP) { + /* OAEP pad the key */ + x = *outlen; + if ((err = pkcs_1_oaep_encode(in, inlen, lparam, + lparamlen, modulus_bitlen, prng, prng_idx, hash_idx, + out, &x)) != CRYPT_OK) { + return err; + } + } else { + /* LTC_PKCS #1 v1.5 pad the key */ + x = *outlen; + if ((err = pkcs_1_v1_5_encode(in, inlen, LTC_LTC_PKCS_1_EME, + modulus_bitlen, prng, prng_idx, + out, &x)) != CRYPT_OK) { + return err; + } + } + + /* rsa exptmod the OAEP or LTC_PKCS #1 v1.5 pad */ + return ltc_mp.rsa_me(out, x, out, outlen, PK_PUBLIC, key); +} +#endif /* LTC_MRSA */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/rsa/rsa_encrypt_key.c,v $ */ +/* $Revision: 1.10 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rsa_exptmod.c + RSA LTC_PKCS exptmod, Tom St Denis + */ + +#ifdef LTC_MRSA + +/** + Compute an RSA modular exponentiation + @param in The input data to send into RSA + @param inlen The length of the input (octets) + @param out [out] The destination + @param outlen [in/out] The max size and resulting size of the output + @param which Which exponent to use, e.g. PK_PRIVATE or PK_PUBLIC + @param key The RSA key to use + @return CRYPT_OK if successful + */ +int rsa_exptmod(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, int which, + rsa_key *key) { + void *tmp, *tmpa, *tmpb; + unsigned long x; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(key != NULL); + + /* is the key of the right type for the operation? */ + if ((which == PK_PRIVATE) && (key->type != PK_PRIVATE)) { + return CRYPT_PK_NOT_PRIVATE; + } + + /* must be a private or public operation */ + if ((which != PK_PRIVATE) && (which != PK_PUBLIC)) { + return CRYPT_PK_INVALID_TYPE; + } + + /* init and copy into tmp */ + if ((err = mp_init_multi(&tmp, &tmpa, &tmpb, NULL)) != CRYPT_OK) { + return err; + } + if ((err = mp_read_unsigned_bin(tmp, (unsigned char *)in, (int)inlen)) != CRYPT_OK) { + goto error; + } + + /* sanity check on the input */ + if (mp_cmp(key->N, tmp) == LTC_MP_LT) { + err = CRYPT_PK_INVALID_SIZE; + goto error; + } + + /* are we using the private exponent and is the key optimized? */ + if (which == PK_PRIVATE) { + /* tmpa = tmp^dP mod p */ + if ((err = mp_exptmod(tmp, key->dP, key->p, tmpa)) != CRYPT_OK) { + goto error; + } + + /* tmpb = tmp^dQ mod q */ + if ((err = mp_exptmod(tmp, key->dQ, key->q, tmpb)) != CRYPT_OK) { + goto error; + } + + /* tmp = (tmpa - tmpb) * qInv (mod p) */ + if ((err = mp_sub(tmpa, tmpb, tmp)) != CRYPT_OK) { + goto error; + } + if ((err = mp_mulmod(tmp, key->qP, key->p, tmp)) != CRYPT_OK) { + goto error; + } + + /* tmp = tmpb + q * tmp */ + if ((err = mp_mul(tmp, key->q, tmp)) != CRYPT_OK) { + goto error; + } + if ((err = mp_add(tmp, tmpb, tmp)) != CRYPT_OK) { + goto error; + } + } else { + /* exptmod it */ + if ((err = mp_exptmod(tmp, key->e, key->N, tmp)) != CRYPT_OK) { + goto error; + } + } + + /* read it back */ + x = (unsigned long)mp_unsigned_bin_size(key->N); + if (x > *outlen) { + *outlen = x; + err = CRYPT_BUFFER_OVERFLOW; + goto error; + } + + /* this should never happen ... */ + if (mp_unsigned_bin_size(tmp) > mp_unsigned_bin_size(key->N)) { + err = CRYPT_ERROR; + goto error; + } + *outlen = x; + + /* convert it */ + zeromem(out, x); + if ((err = mp_to_unsigned_bin(tmp, out + (x - mp_unsigned_bin_size(tmp)))) != CRYPT_OK) { + goto error; + } + + /* clean up and return */ + err = CRYPT_OK; +error: + mp_clear_multi(tmp, tmpa, tmpb, NULL); + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/rsa/rsa_exptmod.c,v $ */ +/* $Revision: 1.18 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rsa_free.c + Free an RSA key, Tom St Denis + */ + +#ifdef LTC_MRSA + +/** + Free an RSA key from memory + @param key The RSA key to free + */ +void rsa_free(rsa_key *key) { + LTC_ARGCHKVD(key != NULL); + mp_clear_multi(key->e, key->d, key->N, key->dQ, key->dP, key->qP, key->p, key->q, NULL); +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/rsa/rsa_free.c,v $ */ +/* $Revision: 1.10 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rsa_import.c + Import a LTC_PKCS RSA key, Tom St Denis + */ + +#ifdef LTC_MRSA + +/** + Import an RSAPublicKey or RSAPrivateKey [two-prime only, only support >= 1024-bit keys, defined in LTC_PKCS #1 v2.1] + @param in The packet to import from + @param inlen It's length (octets) + @param key [out] Destination for newly imported key + @return CRYPT_OK if successful, upon error allocated memory is freed + */ +int rsa_import(const unsigned char *in, unsigned long inlen, rsa_key *key) { + int err; + void *zero; + unsigned char *tmpbuf; + unsigned long t, x, y, z, tmpoid[16]; + ltc_asn1_list ssl_pubkey_hashoid[2]; + ltc_asn1_list ssl_pubkey[2]; + ltc_mp = ltm_desc; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(key != NULL); + LTC_ARGCHK(ltc_mp.name != NULL); + + /* init key */ + if ((err = mp_init_multi(&key->e, &key->d, &key->N, &key->dQ, + &key->dP, &key->qP, &key->p, &key->q, NULL)) != CRYPT_OK) { + return err; + } + + /* see if the OpenSSL DER format RSA public key will work */ + tmpbuf = XCALLOC(1, MAX_RSA_SIZE * 8); + if (tmpbuf == NULL) { + err = CRYPT_MEM; + goto LBL_ERR; + } + + /* this includes the internal hash ID and optional params (NULL in this case) */ + LTC_SET_ASN1(ssl_pubkey_hashoid, 0, LTC_ASN1_OBJECT_IDENTIFIER, tmpoid, sizeof(tmpoid) / sizeof(tmpoid[0])); + LTC_SET_ASN1(ssl_pubkey_hashoid, 1, LTC_ASN1_NULL, NULL, 0); + + /* the actual format of the SSL DER key is odd, it stores a RSAPublicKey in a **BIT** string ... so we have to extract it + then proceed to convert bit to octet + */ + LTC_SET_ASN1(ssl_pubkey, 0, LTC_ASN1_SEQUENCE, &ssl_pubkey_hashoid, 2); + LTC_SET_ASN1(ssl_pubkey, 1, LTC_ASN1_BIT_STRING, tmpbuf, MAX_RSA_SIZE * 8); + + if (der_decode_sequence(in, inlen, + ssl_pubkey, 2UL) == CRYPT_OK) { + /* ok now we have to reassemble the BIT STRING to an OCTET STRING. Thanks OpenSSL... */ + for (t = y = z = x = 0; x < ssl_pubkey[1].size; x++) { + y = (y << 1) | tmpbuf[x]; + if (++z == 8) { + tmpbuf[t++] = (unsigned char)y; + y = 0; + z = 0; + } + } + + /* now it should be SEQUENCE { INTEGER, INTEGER } */ + if ((err = der_decode_sequence_multi(tmpbuf, t, + LTC_ASN1_INTEGER, 1UL, key->N, + LTC_ASN1_INTEGER, 1UL, key->e, + LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { + XFREE(tmpbuf); + goto LBL_ERR; + } + XFREE(tmpbuf); + key->type = PK_PUBLIC; + return CRYPT_OK; + } + XFREE(tmpbuf); + + /* not SSL public key, try to match against LTC_PKCS #1 standards */ + if ((err = der_decode_sequence_multi(in, inlen, + LTC_ASN1_INTEGER, 1UL, key->N, + LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { + goto LBL_ERR; + } + + if (mp_cmp_d(key->N, 0) == LTC_MP_EQ) { + if ((err = mp_init(&zero)) != CRYPT_OK) { + goto LBL_ERR; + } + /* it's a private key */ + if ((err = der_decode_sequence_multi(in, inlen, + LTC_ASN1_INTEGER, 1UL, zero, + LTC_ASN1_INTEGER, 1UL, key->N, + LTC_ASN1_INTEGER, 1UL, key->e, + LTC_ASN1_INTEGER, 1UL, key->d, + LTC_ASN1_INTEGER, 1UL, key->p, + LTC_ASN1_INTEGER, 1UL, key->q, + LTC_ASN1_INTEGER, 1UL, key->dP, + LTC_ASN1_INTEGER, 1UL, key->dQ, + LTC_ASN1_INTEGER, 1UL, key->qP, + LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { + mp_clear(zero); + goto LBL_ERR; + } + mp_clear(zero); + key->type = PK_PRIVATE; + } else if (mp_cmp_d(key->N, 1) == LTC_MP_EQ) { + /* we don't support multi-prime RSA */ + err = CRYPT_PK_INVALID_TYPE; + goto LBL_ERR; + } else { + /* it's a public key and we lack e */ + if ((err = der_decode_sequence_multi(in, inlen, + LTC_ASN1_INTEGER, 1UL, key->N, + LTC_ASN1_INTEGER, 1UL, key->e, + LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { + goto LBL_ERR; + } + key->type = PK_PUBLIC; + } + return CRYPT_OK; +LBL_ERR: + mp_clear_multi(key->d, key->e, key->N, key->dQ, key->dP, key->qP, key->p, key->q, NULL); + return err; +} +#endif /* LTC_MRSA */ + + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/rsa/rsa_import.c,v $ */ +/* $Revision: 1.23 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rsa_make_key.c + RSA key generation, Tom St Denis + */ + +#ifdef LTC_MRSA + +/** + Create an RSA key + @param prng An active PRNG state + @param wprng The index of the PRNG desired + @param size The size of the modulus (key size) desired (octets) + @param e The "e" value (public key). e==65537 is a good choice + @param key [out] Destination of a newly created private key pair + @return CRYPT_OK if successful, upon error all allocated ram is freed + */ +int rsa_make_key(prng_state *prng, int wprng, int size, long e, rsa_key *key) { + void *p, *q, *tmp1, *tmp2, *tmp3; + int err; + + LTC_ARGCHK(ltc_mp.name != NULL); + LTC_ARGCHK(key != NULL); + + if ((size < (MIN_RSA_SIZE / 8)) || (size > (MAX_RSA_SIZE / 8))) { + return CRYPT_INVALID_KEYSIZE; + } + + if ((e < 3) || ((e & 1) == 0)) { + return CRYPT_INVALID_ARG; + } + + if ((err = prng_is_valid(wprng)) != CRYPT_OK) { + return err; + } + + if ((err = mp_init_multi(&p, &q, &tmp1, &tmp2, &tmp3, NULL)) != CRYPT_OK) { + return err; + } + + /* make primes p and q (optimization provided by Wayne Scott) */ + if ((err = mp_set_int(tmp3, e)) != CRYPT_OK) { + goto errkey; + } /* tmp3 = e */ + + /* make prime "p" */ + do { + if ((err = rand_prime(p, size / 2, prng, wprng)) != CRYPT_OK) { + goto errkey; + } + if ((err = mp_sub_d(p, 1, tmp1)) != CRYPT_OK) { + goto errkey; + } /* tmp1 = p-1 */ + if ((err = mp_gcd(tmp1, tmp3, tmp2)) != CRYPT_OK) { + goto errkey; + } /* tmp2 = gcd(p-1, e) */ + } while (mp_cmp_d(tmp2, 1) != 0); /* while e divides p-1 */ + + /* make prime "q" */ + do { + if ((err = rand_prime(q, size / 2, prng, wprng)) != CRYPT_OK) { + goto errkey; + } + if ((err = mp_sub_d(q, 1, tmp1)) != CRYPT_OK) { + goto errkey; + } /* tmp1 = q-1 */ + if ((err = mp_gcd(tmp1, tmp3, tmp2)) != CRYPT_OK) { + goto errkey; + } /* tmp2 = gcd(q-1, e) */ + } while (mp_cmp_d(tmp2, 1) != 0); /* while e divides q-1 */ + + /* tmp1 = lcm(p-1, q-1) */ + if ((err = mp_sub_d(p, 1, tmp2)) != CRYPT_OK) { + goto errkey; + } /* tmp2 = p-1 */ + /* tmp1 = q-1 (previous do/while loop) */ + if ((err = mp_lcm(tmp1, tmp2, tmp1)) != CRYPT_OK) { + goto errkey; + } /* tmp1 = lcm(p-1, q-1) */ + + /* make key */ + if ((err = mp_init_multi(&key->e, &key->d, &key->N, &key->dQ, &key->dP, &key->qP, &key->p, &key->q, NULL)) != CRYPT_OK) { + goto errkey; + } + + if ((err = mp_set_int(key->e, e)) != CRYPT_OK) { + goto errkey; + } /* key->e = e */ + if ((err = mp_invmod(key->e, tmp1, key->d)) != CRYPT_OK) { + goto errkey; + } /* key->d = 1/e mod lcm(p-1,q-1) */ + if ((err = mp_mul(p, q, key->N)) != CRYPT_OK) { + goto errkey; + } /* key->N = pq */ + + /* optimize for CRT now */ + /* find d mod q-1 and d mod p-1 */ + if ((err = mp_sub_d(p, 1, tmp1)) != CRYPT_OK) { + goto errkey; + } /* tmp1 = q-1 */ + if ((err = mp_sub_d(q, 1, tmp2)) != CRYPT_OK) { + goto errkey; + } /* tmp2 = p-1 */ + if ((err = mp_mod(key->d, tmp1, key->dP)) != CRYPT_OK) { + goto errkey; + } /* dP = d mod p-1 */ + if ((err = mp_mod(key->d, tmp2, key->dQ)) != CRYPT_OK) { + goto errkey; + } /* dQ = d mod q-1 */ + if ((err = mp_invmod(q, p, key->qP)) != CRYPT_OK) { + goto errkey; + } /* qP = 1/q mod p */ + + if ((err = mp_copy(p, key->p)) != CRYPT_OK) { + goto errkey; + } + if ((err = mp_copy(q, key->q)) != CRYPT_OK) { + goto errkey; + } + + /* set key type (in this case it's CRT optimized) */ + key->type = PK_PRIVATE; + + /* return ok and free temps */ + err = CRYPT_OK; + goto cleanup; +errkey: + mp_clear_multi(key->d, key->e, key->N, key->dQ, key->dP, key->qP, key->p, key->q, NULL); +cleanup: + mp_clear_multi(tmp3, tmp2, tmp1, p, q, NULL); + return err; +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/rsa/rsa_make_key.c,v $ */ +/* $Revision: 1.16 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rsa_sign_hash.c + RSA LTC_PKCS #1 v1.5 and v2 PSS sign hash, Tom St Denis and Andreas Lange + */ + +#ifdef LTC_MRSA + +/** + LTC_PKCS #1 pad then sign + @param in The hash to sign + @param inlen The length of the hash to sign (octets) + @param out [out] The signature + @param outlen [in/out] The max size and resulting size of the signature + @param padding Type of padding (LTC_LTC_PKCS_1_PSS or LTC_LTC_PKCS_1_V1_5) + @param prng An active PRNG state + @param prng_idx The index of the PRNG desired + @param hash_idx The index of the hash desired + @param saltlen The length of the salt desired (octets) + @param key The private RSA key to use + @return CRYPT_OK if successful + */ +int rsa_sign_hash_ex(const unsigned char *in, unsigned long inlen, + unsigned char *out, unsigned long *outlen, + int padding, + prng_state *prng, int prng_idx, + int hash_idx, unsigned long saltlen, + rsa_key *key) { + unsigned long modulus_bitlen, modulus_bytelen, x, y; + int err; + + LTC_ARGCHK(in != NULL); + LTC_ARGCHK(out != NULL); + LTC_ARGCHK(outlen != NULL); + LTC_ARGCHK(key != NULL); + + /* valid padding? */ + if ((padding != LTC_LTC_PKCS_1_V1_5) && (padding != LTC_LTC_PKCS_1_PSS)) { + return CRYPT_PK_INVALID_PADDING; + } + + hash_idx = register_hash(&sha256_desc); + + if (padding == LTC_LTC_PKCS_1_PSS) { + /* valid prng and hash ? */ + if ((err = prng_is_valid(prng_idx)) != CRYPT_OK) { + return err; + } + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + } + + /* get modulus len in bits */ + modulus_bitlen = mp_count_bits((key->N)); + + /* outlen must be at least the size of the modulus */ + modulus_bytelen = mp_unsigned_bin_size((key->N)); + if (modulus_bytelen > *outlen) { + *outlen = modulus_bytelen; + return CRYPT_BUFFER_OVERFLOW; + } + + if (padding == LTC_LTC_PKCS_1_PSS) { + /* PSS pad the key */ + x = *outlen; + if ((err = pkcs_1_pss_encode(in, inlen, saltlen, prng, prng_idx, + hash_idx, modulus_bitlen, out, &x)) != CRYPT_OK) { + return err; + } + } else { + /* LTC_PKCS #1 v1.5 pad the hash */ + unsigned char *tmpin; + ltc_asn1_list digestinfo[2], siginfo[2]; + + /* not all hashes have OIDs... so sad */ + if (hash_descriptor[hash_idx].OIDlen == 0) { + return CRYPT_INVALID_ARG; + } + + /* construct the SEQUENCE + SEQUENCE { + SEQUENCE {hashoid OID + blah NULL + } + hash OCTET STRING + } + */ + LTC_SET_ASN1(digestinfo, 0, LTC_ASN1_OBJECT_IDENTIFIER, hash_descriptor[hash_idx].OID, hash_descriptor[hash_idx].OIDlen); + LTC_SET_ASN1(digestinfo, 1, LTC_ASN1_NULL, NULL, 0); + LTC_SET_ASN1(siginfo, 0, LTC_ASN1_SEQUENCE, digestinfo, 2); + LTC_SET_ASN1(siginfo, 1, LTC_ASN1_OCTET_STRING, in, inlen); + + /* allocate memory for the encoding */ + y = mp_unsigned_bin_size(key->N); + tmpin = XMALLOC(y); + if (tmpin == NULL) { + return CRYPT_MEM; + } + + if ((err = der_encode_sequence(siginfo, 2, tmpin, &y)) != CRYPT_OK) { + XFREE(tmpin); + return err; + } + + x = *outlen; + if ((err = pkcs_1_v1_5_encode(tmpin, y, LTC_LTC_PKCS_1_EMSA, + modulus_bitlen, NULL, 0, + out, &x)) != CRYPT_OK) { + XFREE(tmpin); + return err; + } + XFREE(tmpin); + } + + /* RSA encode it */ + return ltc_mp.rsa_me(out, x, out, outlen, PK_PRIVATE, key); +} +#endif /* LTC_MRSA */ + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/rsa/rsa_sign_hash.c,v $ */ +/* $Revision: 1.11 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file rsa_verify_hash.c + RSA LTC_PKCS #1 v1.5 or v2 PSS signature verification, Tom St Denis and Andreas Lange + */ + +#ifdef LTC_MRSA + +/** + LTC_PKCS #1 de-sign then v1.5 or PSS depad + @param sig The signature data + @param siglen The length of the signature data (octets) + @param hash The hash of the message that was signed + @param hashlen The length of the hash of the message that was signed (octets) + @param padding Type of padding (LTC_LTC_PKCS_1_PSS or LTC_LTC_PKCS_1_V1_5) + @param hash_idx The index of the desired hash + @param saltlen The length of the salt used during signature + @param stat [out] The result of the signature comparison, 1==valid, 0==invalid + @param key The public RSA key corresponding to the key that performed the signature + @return CRYPT_OK on success (even if the signature is invalid) + */ +int rsa_verify_hash_ex(const unsigned char *sig, unsigned long siglen, + const unsigned char *hash, unsigned long hashlen, + int padding, + int hash_idx, unsigned long saltlen, + int *stat, rsa_key *key) { + unsigned long modulus_bitlen, modulus_bytelen, x; + int err; + unsigned char *tmpbuf; + + LTC_ARGCHK(hash != NULL); + LTC_ARGCHK(sig != NULL); + LTC_ARGCHK(stat != NULL); + LTC_ARGCHK(key != NULL); + + /* default to invalid */ + *stat = 0; + + /* valid padding? */ + + if ((padding != LTC_LTC_PKCS_1_V1_5) && + (padding != LTC_LTC_PKCS_1_PSS)) { + return CRYPT_PK_INVALID_PADDING; + } + + hash_idx = register_hash(&sha256_desc); + + if (padding == LTC_LTC_PKCS_1_PSS) { + /* valid hash ? */ + if ((err = hash_is_valid(hash_idx)) != CRYPT_OK) { + return err; + } + } + + /* get modulus len in bits */ + modulus_bitlen = mp_count_bits((key->N)); + + /* outlen must be at least the size of the modulus */ + modulus_bytelen = mp_unsigned_bin_size((key->N)); + if (modulus_bytelen != siglen) { + return CRYPT_INVALID_PACKET; + } + + /* allocate temp buffer for decoded sig */ + tmpbuf = XMALLOC(siglen); + if (tmpbuf == NULL) { + return CRYPT_MEM; + } + + /* RSA decode it */ + x = siglen; + if ((err = ltc_mp.rsa_me(sig, siglen, tmpbuf, &x, PK_PUBLIC, key)) != CRYPT_OK) { + XFREE(tmpbuf); + return err; + } + + /* make sure the output is the right size */ + if (x != siglen) { + XFREE(tmpbuf); + return CRYPT_INVALID_PACKET; + } + + if (padding == LTC_LTC_PKCS_1_PSS) { + /* PSS decode and verify it */ + err = pkcs_1_pss_decode(hash, hashlen, tmpbuf, x, saltlen, hash_idx, modulus_bitlen, stat); + } else { + /* LTC_PKCS #1 v1.5 decode it */ + unsigned char *out; + unsigned long outlen, loid[16]; + int decoded; + ltc_asn1_list digestinfo[2], siginfo[2]; + + /* not all hashes have OIDs... so sad */ + if (hash_descriptor[hash_idx].OIDlen == 0) { + err = CRYPT_INVALID_ARG; + goto bail_2; + } + + /* allocate temp buffer for decoded hash */ + outlen = ((modulus_bitlen >> 3) + (modulus_bitlen & 7 ? 1 : 0)) - 3; + out = XMALLOC(outlen); + if (out == NULL) { + err = CRYPT_MEM; + goto bail_2; + } + + if ((err = pkcs_1_v1_5_decode(tmpbuf, x, LTC_LTC_PKCS_1_EMSA, modulus_bitlen, out, &outlen, &decoded)) != CRYPT_OK) { + XFREE(out); + goto bail_2; + } + + /* now we must decode out[0...outlen-1] using ASN.1, test the OID and then test the hash */ + + /* construct the SEQUENCE + SEQUENCE { + SEQUENCE {hashoid OID + blah NULL + } + hash OCTET STRING + } + */ + LTC_SET_ASN1(digestinfo, 0, LTC_ASN1_OBJECT_IDENTIFIER, loid, sizeof(loid) / sizeof(loid[0])); + LTC_SET_ASN1(digestinfo, 1, LTC_ASN1_NULL, NULL, 0); + LTC_SET_ASN1(siginfo, 0, LTC_ASN1_SEQUENCE, digestinfo, 2); + LTC_SET_ASN1(siginfo, 1, LTC_ASN1_OCTET_STRING, tmpbuf, siglen); + + if ((err = der_decode_sequence(out, outlen, siginfo, 2)) != CRYPT_OK) { + XFREE(out); + goto bail_2; + } + + /* test OID */ + if ((digestinfo[0].size == hash_descriptor[hash_idx].OIDlen) && + (XMEMCMP(digestinfo[0].data, hash_descriptor[hash_idx].OID, sizeof(unsigned long) * hash_descriptor[hash_idx].OIDlen) == 0) && + (siginfo[1].size == hashlen) && + (XMEMCMP(siginfo[1].data, hash, hashlen) == 0)) { + *stat = 1; + } + + #ifdef LTC_CLEAN_STACK + zeromem(out, outlen); + #endif + XFREE(out); + } + +bail_2: + #ifdef LTC_CLEAN_STACK + zeromem(tmpbuf, siglen); + #endif + XFREE(tmpbuf); + return err; +} +#endif /* LTC_MRSA */ + +int rsa_create_signature(unsigned char *sig, unsigned long* siglen, + const unsigned char *hash, unsigned long hashlen, + rsa_key *key) { + return rsa_sign_hash_ex(hash, hashlen, sig, siglen, LTC_LTC_PKCS_1_V1_5, NULL, 0, 0, 0, key); +} + +int rsa_verify_signature(const unsigned char *sig, unsigned long siglen, + const unsigned char *hash, unsigned long hashlen, + int* stat, rsa_key *key) { + return rsa_verify_hash_ex(sig, siglen, hash, hashlen, LTC_LTC_PKCS_1_V1_5, 0, 0, stat, key); +} + +/* $Source: /cvs/libtom/libtomcrypt/src/pk/rsa/rsa_verify_hash.c,v $ */ +/* $Revision: 1.13 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file sprng.c + Secure PRNG, Tom St Denis + */ + +/* A secure PRNG using the RNG functions. Basically this is a + * wrapper that allows you to use a secure RNG as a PRNG + * in the various other functions. + */ + +#ifdef LTC_SPRNG + +const struct ltc_prng_descriptor sprng_desc = +{ + "sprng", 0, + &sprng_start, + &sprng_add_entropy, + &sprng_ready, + &sprng_read, + &sprng_done, + &sprng_export, + &sprng_import, + &sprng_test +}; + +/** + Start the PRNG + @param prng [out] The PRNG state to initialize + @return CRYPT_OK if successful + */ +int sprng_start(prng_state *prng) { + return CRYPT_OK; +} + +/** + Add entropy to the PRNG state + @param in The data to add + @param inlen Length of the data to add + @param prng PRNG state to update + @return CRYPT_OK if successful + */ +int sprng_add_entropy(const unsigned char *in, unsigned long inlen, prng_state *prng) { + return CRYPT_OK; +} + +/** + Make the PRNG ready to read from + @param prng The PRNG to make active + @return CRYPT_OK if successful + */ +int sprng_ready(prng_state *prng) { + return CRYPT_OK; +} + +/** + Read from the PRNG + @param out Destination + @param outlen Length of output + @param prng The active PRNG to read from + @return Number of octets read + */ +unsigned long sprng_read(unsigned char *out, unsigned long outlen, prng_state *prng) { + LTC_ARGCHK(out != NULL); + return rng_get_bytes(out, outlen, NULL); +} + +/** + Terminate the PRNG + @param prng The PRNG to terminate + @return CRYPT_OK if successful + */ +int sprng_done(prng_state *prng) { + return CRYPT_OK; +} + +/** + Export the PRNG state + @param out [out] Destination + @param outlen [in/out] Max size and resulting size of the state + @param prng The PRNG to export + @return CRYPT_OK if successful + */ +int sprng_export(unsigned char *out, unsigned long *outlen, prng_state *prng) { + LTC_ARGCHK(outlen != NULL); + + *outlen = 0; + return CRYPT_OK; +} + +/** + Import a PRNG state + @param in The PRNG state + @param inlen Size of the state + @param prng The PRNG to import + @return CRYPT_OK if successful + */ +int sprng_import(const unsigned char *in, unsigned long inlen, prng_state *prng) { + return CRYPT_OK; +} + +/** + PRNG self-test + @return CRYPT_OK if successful, CRYPT_NOP if self-testing has been disabled + */ +int sprng_test(void) { + return CRYPT_OK; +} +#endif + + + +/* $Source: /cvs/libtom/libtomcrypt/src/prngs/sprng.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file zeromem.c + Zero a block of memory, Tom St Denis + */ + +/** + Zero a block of memory + @param out The destination of the area to zero + @param outlen The length of the area to zero (octets) + */ +void zeromem(void *out, size_t outlen) { + unsigned char *mem = out; + + LTC_ARGCHKVD(out != NULL); + while (outlen-- > 0) { + *mem++ = 0; + } +} + +/* $Source: /cvs/libtom/libtomcrypt/src/misc/zeromem.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file sha1.c + LTC_SHA1 code by Tom St Denis + */ + + +#ifdef LTC_SHA1 + +const struct ltc_hash_descriptor sha1_desc = +{ + "sha1", + 2, + 20, + 64, + + /* OID */ + { 1, 3, 14, 3, 2, 26, }, + 6, + + &sha1_init, + &sha1_process, + &sha1_done, + &sha1_test, + NULL +}; + + #define F0(x, y, z) (z ^ (x & (y ^ z))) + #define F1(x, y, z) (x ^ y ^ z) + #define F2(x, y, z) ((x & y) | (z & (x | y))) + #define F3(x, y, z) (x ^ y ^ z) + + #ifdef LTC_CLEAN_STACK +static int _sha1_compress(hash_state *md, unsigned char *buf) + #else +static int sha1_compress(hash_state *md, unsigned char *buf) + #endif +{ + ulong32 a, b, c, d, e, W[80], i; + + #ifdef LTC_SMALL_CODE + ulong32 t; + #endif + + /* copy the state into 512-bits into W[0..15] */ + for (i = 0; i < 16; i++) { + LOAD32H(W[i], buf + (4 * i)); + } + + /* copy state */ + a = md->sha1.state[0]; + b = md->sha1.state[1]; + c = md->sha1.state[2]; + d = md->sha1.state[3]; + e = md->sha1.state[4]; + + /* expand it */ + for (i = 16; i < 80; i++) { + W[i] = ROL(W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16], 1); + } + + /* compress */ + /* round one */ + #define FF0(a, b, c, d, e, i) e = (ROLc(a, 5) + F0(b, c, d) + e + W[i] + 0x5a827999UL); b = ROLc(b, 30); + #define FF1(a, b, c, d, e, i) e = (ROLc(a, 5) + F1(b, c, d) + e + W[i] + 0x6ed9eba1UL); b = ROLc(b, 30); + #define FF2(a, b, c, d, e, i) e = (ROLc(a, 5) + F2(b, c, d) + e + W[i] + 0x8f1bbcdcUL); b = ROLc(b, 30); + #define FF3(a, b, c, d, e, i) e = (ROLc(a, 5) + F3(b, c, d) + e + W[i] + 0xca62c1d6UL); b = ROLc(b, 30); + + #ifdef LTC_SMALL_CODE + for (i = 0; i < 20; ) { + FF0(a, b, c, d, e, i++); + t = e; + e = d; + d = c; + c = b; + b = a; + a = t; + } + + for ( ; i < 40; ) { + FF1(a, b, c, d, e, i++); + t = e; + e = d; + d = c; + c = b; + b = a; + a = t; + } + + for ( ; i < 60; ) { + FF2(a, b, c, d, e, i++); + t = e; + e = d; + d = c; + c = b; + b = a; + a = t; + } + + for ( ; i < 80; ) { + FF3(a, b, c, d, e, i++); + t = e; + e = d; + d = c; + c = b; + b = a; + a = t; + } + + #else + for (i = 0; i < 20; ) { + FF0(a, b, c, d, e, i++); + FF0(e, a, b, c, d, i++); + FF0(d, e, a, b, c, i++); + FF0(c, d, e, a, b, i++); + FF0(b, c, d, e, a, i++); + } + + /* round two */ + for ( ; i < 40; ) { + FF1(a, b, c, d, e, i++); + FF1(e, a, b, c, d, i++); + FF1(d, e, a, b, c, i++); + FF1(c, d, e, a, b, i++); + FF1(b, c, d, e, a, i++); + } + + /* round three */ + for ( ; i < 60; ) { + FF2(a, b, c, d, e, i++); + FF2(e, a, b, c, d, i++); + FF2(d, e, a, b, c, i++); + FF2(c, d, e, a, b, i++); + FF2(b, c, d, e, a, i++); + } + + /* round four */ + for ( ; i < 80; ) { + FF3(a, b, c, d, e, i++); + FF3(e, a, b, c, d, i++); + FF3(d, e, a, b, c, i++); + FF3(c, d, e, a, b, i++); + FF3(b, c, d, e, a, i++); + } + #endif + + #undef FF0 + #undef FF1 + #undef FF2 + #undef FF3 + + /* store */ + md->sha1.state[0] = md->sha1.state[0] + a; + md->sha1.state[1] = md->sha1.state[1] + b; + md->sha1.state[2] = md->sha1.state[2] + c; + md->sha1.state[3] = md->sha1.state[3] + d; + md->sha1.state[4] = md->sha1.state[4] + e; + + return CRYPT_OK; +} + + #ifdef LTC_CLEAN_STACK +static int sha1_compress(hash_state *md, unsigned char *buf) { + int err; + + err = _sha1_compress(md, buf); + burn_stack(sizeof(ulong32) * 87); + return err; +} + #endif + +/** + Initialize the hash state + @param md The hash state you wish to initialize + @return CRYPT_OK if successful + */ +int sha1_init(hash_state *md) { + LTC_ARGCHK(md != NULL); + md->sha1.state[0] = 0x67452301UL; + md->sha1.state[1] = 0xefcdab89UL; + md->sha1.state[2] = 0x98badcfeUL; + md->sha1.state[3] = 0x10325476UL; + md->sha1.state[4] = 0xc3d2e1f0UL; + md->sha1.curlen = 0; + md->sha1.length = 0; + return CRYPT_OK; +} + +/** + Process a block of memory though the hash + @param md The hash state + @param in The data to hash + @param inlen The length of the data (octets) + @return CRYPT_OK if successful + */ +HASH_PROCESS(sha1_process, sha1_compress, sha1, 64) + +/** + Terminate the hash to get the digest + @param md The hash state + @param out [out] The destination of the hash (20 bytes) + @return CRYPT_OK if successful + */ +int sha1_done(hash_state *md, unsigned char *out) { + int i; + + LTC_ARGCHK(md != NULL); + LTC_ARGCHK(out != NULL); + + if (md->sha1.curlen >= sizeof(md->sha1.buf)) { + return CRYPT_INVALID_ARG; + } + + /* increase the length of the message */ + md->sha1.length += md->sha1.curlen * 8; + + /* append the '1' bit */ + md->sha1.buf[md->sha1.curlen++] = (unsigned char)0x80; + + /* if the length is currently above 56 bytes we append zeros + * then compress. Then we can fall back to padding zeros and length + * encoding like normal. + */ + if (md->sha1.curlen > 56) { + while (md->sha1.curlen < 64) { + md->sha1.buf[md->sha1.curlen++] = (unsigned char)0; + } + sha1_compress(md, md->sha1.buf); + md->sha1.curlen = 0; + } + + /* pad upto 56 bytes of zeroes */ + while (md->sha1.curlen < 56) { + md->sha1.buf[md->sha1.curlen++] = (unsigned char)0; + } + + /* store length */ + STORE64H(md->sha1.length, md->sha1.buf + 56); + sha1_compress(md, md->sha1.buf); + + /* copy output */ + for (i = 0; i < 5; i++) { + STORE32H(md->sha1.state[i], out + (4 * i)); + } + #ifdef LTC_CLEAN_STACK + zeromem(md, sizeof(hash_state)); + #endif + return CRYPT_OK; +} + +/** + Self-test the hash + @return CRYPT_OK if successful, CRYPT_NOP if self-tests have been disabled + */ +int sha1_test(void) { + #ifndef LTC_TEST + return CRYPT_NOP; + #else + static const struct { + char *msg; + unsigned char hash[20]; + } tests[] = { + { "abc", + { 0xa9, 0x99, 0x3e, 0x36, 0x47, 0x06, 0x81, 0x6a, + 0xba, 0x3e, 0x25, 0x71, 0x78, 0x50, 0xc2, 0x6c, + 0x9c, 0xd0, 0xd8, 0x9d } }, + { "abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq", + { 0x84, 0x98, 0x3E, 0x44, 0x1C, 0x3B, 0xD2, 0x6E, + 0xBA, 0xAE, 0x4A, 0xA1, 0xF9, 0x51, 0x29, 0xE5, + 0xE5, 0x46, 0x70, 0xF1 } } + }; + + int i; + unsigned char tmp[20]; + hash_state md; + + for (i = 0; i < (int)(sizeof(tests) / sizeof(tests[0])); i++) { + sha1_init(&md); + sha1_process(&md, (unsigned char *)tests[i].msg, (unsigned long)strlen(tests[i].msg)); + sha1_done(&md, tmp); + if (XMEMCMP(tmp, tests[i].hash, 20) != 0) { + return CRYPT_FAIL_TESTVECTOR; + } + } + return CRYPT_OK; + #endif +} +#endif + + + +/* $Source: /cvs/libtom/libtomcrypt/src/hashes/sha1.c,v $ */ +/* $Revision: 1.10 $ */ +/* $Date: 2007/05/12 14:25:28 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file sha256.c + LTC_SHA256 by Tom St Denis +*/ + +#ifdef LTC_SHA256 + +const struct ltc_hash_descriptor sha256_desc = +{ + "sha256", + 0, + 32, + 64, + + /* OID */ + { 2, 16, 840, 1, 101, 3, 4, 2, 1, }, + 9, + + &sha256_init, + &sha256_process, + &sha256_done, + &sha256_test, + NULL +}; + +#ifdef LTC_SMALL_CODE +/* the K array */ +static const ulong32 K[64] = { + 0x428a2f98UL, 0x71374491UL, 0xb5c0fbcfUL, 0xe9b5dba5UL, 0x3956c25bUL, + 0x59f111f1UL, 0x923f82a4UL, 0xab1c5ed5UL, 0xd807aa98UL, 0x12835b01UL, + 0x243185beUL, 0x550c7dc3UL, 0x72be5d74UL, 0x80deb1feUL, 0x9bdc06a7UL, + 0xc19bf174UL, 0xe49b69c1UL, 0xefbe4786UL, 0x0fc19dc6UL, 0x240ca1ccUL, + 0x2de92c6fUL, 0x4a7484aaUL, 0x5cb0a9dcUL, 0x76f988daUL, 0x983e5152UL, + 0xa831c66dUL, 0xb00327c8UL, 0xbf597fc7UL, 0xc6e00bf3UL, 0xd5a79147UL, + 0x06ca6351UL, 0x14292967UL, 0x27b70a85UL, 0x2e1b2138UL, 0x4d2c6dfcUL, + 0x53380d13UL, 0x650a7354UL, 0x766a0abbUL, 0x81c2c92eUL, 0x92722c85UL, + 0xa2bfe8a1UL, 0xa81a664bUL, 0xc24b8b70UL, 0xc76c51a3UL, 0xd192e819UL, + 0xd6990624UL, 0xf40e3585UL, 0x106aa070UL, 0x19a4c116UL, 0x1e376c08UL, + 0x2748774cUL, 0x34b0bcb5UL, 0x391c0cb3UL, 0x4ed8aa4aUL, 0x5b9cca4fUL, + 0x682e6ff3UL, 0x748f82eeUL, 0x78a5636fUL, 0x84c87814UL, 0x8cc70208UL, + 0x90befffaUL, 0xa4506cebUL, 0xbef9a3f7UL, 0xc67178f2UL +}; +#endif + +/* Various logical functions */ +#define Ch(x,y,z) (z ^ (x & (y ^ z))) +#define Maj(x,y,z) (((x | y) & z) | (x & y)) +#define S(x, n) RORc((x),(n)) +#define R(x, n) (((x)&0xFFFFFFFFUL)>>(n)) +#define Sigma0(x) (S(x, 2) ^ S(x, 13) ^ S(x, 22)) +#define Sigma1(x) (S(x, 6) ^ S(x, 11) ^ S(x, 25)) +#define Gamma0(x) (S(x, 7) ^ S(x, 18) ^ R(x, 3)) +#define Gamma1(x) (S(x, 17) ^ S(x, 19) ^ R(x, 10)) + +/* compress 512-bits */ +#ifdef LTC_CLEAN_STACK +static int _sha256_compress(hash_state * md, unsigned char *buf) +#else +static int sha256_compress(hash_state * md, unsigned char *buf) +#endif +{ + ulong32 S[8], W[64], t0, t1; +#ifdef LTC_SMALL_CODE + ulong32 t; +#endif + int i; + + /* copy state into S */ + for (i = 0; i < 8; i++) { + S[i] = md->sha256.state[i]; + } + + /* copy the state into 512-bits into W[0..15] */ + for (i = 0; i < 16; i++) { + LOAD32H(W[i], buf + (4*i)); + } + + /* fill W[16..63] */ + for (i = 16; i < 64; i++) { + W[i] = Gamma1(W[i - 2]) + W[i - 7] + Gamma0(W[i - 15]) + W[i - 16]; + } + + /* Compress */ +#ifdef LTC_SMALL_CODE +#define RND(a,b,c,d,e,f,g,h,i) \ + t0 = h + Sigma1(e) + Ch(e, f, g) + K[i] + W[i]; \ + t1 = Sigma0(a) + Maj(a, b, c); \ + d += t0; \ + h = t0 + t1; + + for (i = 0; i < 64; ++i) { + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],i); + t = S[7]; S[7] = S[6]; S[6] = S[5]; S[5] = S[4]; + S[4] = S[3]; S[3] = S[2]; S[2] = S[1]; S[1] = S[0]; S[0] = t; + } +#else +#define RND(a,b,c,d,e,f,g,h,i,ki) \ + t0 = h + Sigma1(e) + Ch(e, f, g) + ki + W[i]; \ + t1 = Sigma0(a) + Maj(a, b, c); \ + d += t0; \ + h = t0 + t1; + + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],0,0x428a2f98); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],1,0x71374491); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],2,0xb5c0fbcf); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],3,0xe9b5dba5); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],4,0x3956c25b); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],5,0x59f111f1); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],6,0x923f82a4); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],7,0xab1c5ed5); + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],8,0xd807aa98); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],9,0x12835b01); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],10,0x243185be); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],11,0x550c7dc3); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],12,0x72be5d74); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],13,0x80deb1fe); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],14,0x9bdc06a7); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],15,0xc19bf174); + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],16,0xe49b69c1); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],17,0xefbe4786); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],18,0x0fc19dc6); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],19,0x240ca1cc); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],20,0x2de92c6f); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],21,0x4a7484aa); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],22,0x5cb0a9dc); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],23,0x76f988da); + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],24,0x983e5152); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],25,0xa831c66d); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],26,0xb00327c8); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],27,0xbf597fc7); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],28,0xc6e00bf3); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],29,0xd5a79147); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],30,0x06ca6351); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],31,0x14292967); + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],32,0x27b70a85); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],33,0x2e1b2138); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],34,0x4d2c6dfc); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],35,0x53380d13); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],36,0x650a7354); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],37,0x766a0abb); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],38,0x81c2c92e); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],39,0x92722c85); + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],40,0xa2bfe8a1); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],41,0xa81a664b); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],42,0xc24b8b70); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],43,0xc76c51a3); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],44,0xd192e819); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],45,0xd6990624); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],46,0xf40e3585); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],47,0x106aa070); + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],48,0x19a4c116); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],49,0x1e376c08); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],50,0x2748774c); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],51,0x34b0bcb5); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],52,0x391c0cb3); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],53,0x4ed8aa4a); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],54,0x5b9cca4f); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],55,0x682e6ff3); + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],56,0x748f82ee); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],57,0x78a5636f); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],58,0x84c87814); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],59,0x8cc70208); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],60,0x90befffa); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],61,0xa4506ceb); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],62,0xbef9a3f7); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],63,0xc67178f2); + +#undef RND + +#endif + + /* feedback */ + for (i = 0; i < 8; i++) { + md->sha256.state[i] = md->sha256.state[i] + S[i]; + } + return CRYPT_OK; +} + +#ifdef LTC_CLEAN_STACK +static int sha256_compress(hash_state * md, unsigned char *buf) +{ + int err; + err = _sha256_compress(md, buf); + burn_stack(sizeof(ulong32) * 74); + return err; +} +#endif + +/** + Initialize the hash state + @param md The hash state you wish to initialize + @return CRYPT_OK if successful +*/ +int sha256_init(hash_state * md) +{ + LTC_ARGCHK(md != NULL); + + md->sha256.curlen = 0; + md->sha256.length = 0; + md->sha256.state[0] = 0x6A09E667UL; + md->sha256.state[1] = 0xBB67AE85UL; + md->sha256.state[2] = 0x3C6EF372UL; + md->sha256.state[3] = 0xA54FF53AUL; + md->sha256.state[4] = 0x510E527FUL; + md->sha256.state[5] = 0x9B05688CUL; + md->sha256.state[6] = 0x1F83D9ABUL; + md->sha256.state[7] = 0x5BE0CD19UL; + return CRYPT_OK; +} + +/** + Process a block of memory though the hash + @param md The hash state + @param in The data to hash + @param inlen The length of the data (octets) + @return CRYPT_OK if successful +*/ +HASH_PROCESS(sha256_process, sha256_compress, sha256, 64) + +/** + Terminate the hash to get the digest + @param md The hash state + @param out [out] The destination of the hash (32 bytes) + @return CRYPT_OK if successful +*/ +int sha256_done(hash_state * md, unsigned char *out) +{ + int i; + + LTC_ARGCHK(md != NULL); + LTC_ARGCHK(out != NULL); + + if (md->sha256.curlen >= sizeof(md->sha256.buf)) { + return CRYPT_INVALID_ARG; + } + + + /* increase the length of the message */ + md->sha256.length += md->sha256.curlen * 8; + + /* append the '1' bit */ + md->sha256.buf[md->sha256.curlen++] = (unsigned char)0x80; + + /* if the length is currently above 56 bytes we append zeros + * then compress. Then we can fall back to padding zeros and length + * encoding like normal. + */ + if (md->sha256.curlen > 56) { + while (md->sha256.curlen < 64) { + md->sha256.buf[md->sha256.curlen++] = (unsigned char)0; + } + sha256_compress(md, md->sha256.buf); + md->sha256.curlen = 0; + } + + /* pad upto 56 bytes of zeroes */ + while (md->sha256.curlen < 56) { + md->sha256.buf[md->sha256.curlen++] = (unsigned char)0; + } + + /* store length */ + STORE64H(md->sha256.length, md->sha256.buf+56); + sha256_compress(md, md->sha256.buf); + + /* copy output */ + for (i = 0; i < 8; i++) { + STORE32H(md->sha256.state[i], out+(4*i)); + } +#ifdef LTC_CLEAN_STACK + zeromem(md, sizeof(hash_state)); +#endif + return CRYPT_OK; +} + +/** + Self-test the hash + @return CRYPT_OK if successful, CRYPT_NOP if self-tests have been disabled +*/ +int sha256_test(void) +{ + #ifndef LTC_TEST + return CRYPT_NOP; + #else + static const struct { + char *msg; + unsigned char hash[32]; + } tests[] = { + { "abc", + { 0xba, 0x78, 0x16, 0xbf, 0x8f, 0x01, 0xcf, 0xea, + 0x41, 0x41, 0x40, 0xde, 0x5d, 0xae, 0x22, 0x23, + 0xb0, 0x03, 0x61, 0xa3, 0x96, 0x17, 0x7a, 0x9c, + 0xb4, 0x10, 0xff, 0x61, 0xf2, 0x00, 0x15, 0xad } + }, + { "abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq", + { 0x24, 0x8d, 0x6a, 0x61, 0xd2, 0x06, 0x38, 0xb8, + 0xe5, 0xc0, 0x26, 0x93, 0x0c, 0x3e, 0x60, 0x39, + 0xa3, 0x3c, 0xe4, 0x59, 0x64, 0xff, 0x21, 0x67, + 0xf6, 0xec, 0xed, 0xd4, 0x19, 0xdb, 0x06, 0xc1 } + }, + }; + + int i; + unsigned char tmp[32]; + hash_state md; + + for (i = 0; i < (int)(sizeof(tests) / sizeof(tests[0])); i++) { + sha256_init(&md); + sha256_process(&md, (unsigned char*)tests[i].msg, (unsigned long)strlen(tests[i].msg)); + sha256_done(&md, tmp); + if (XMEMCMP(tmp, tests[i].hash, 32) != 0) { + return CRYPT_FAIL_TESTVECTOR; + } + } + return CRYPT_OK; + #endif +} + +#endif + + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ +/** + @param sha384.c + LTC_SHA384 hash included in sha512.c, Tom St Denis +*/ + + + +#if defined(LTC_SHA384) && defined(LTC_SHA512) + +const struct ltc_hash_descriptor sha384_desc = +{ + "sha384", + 4, + 48, + 128, + + /* OID */ + { 2, 16, 840, 1, 101, 3, 4, 2, 2, }, + 9, + + &sha384_init, + &sha512_process, + &sha384_done, + &sha384_test, + NULL +}; + +/** + Initialize the hash state + @param md The hash state you wish to initialize + @return CRYPT_OK if successful +*/ +int sha384_init(hash_state * md) +{ + LTC_ARGCHK(md != NULL); + + md->sha512.curlen = 0; + md->sha512.length = 0; + md->sha512.state[0] = CONST64(0xcbbb9d5dc1059ed8); + md->sha512.state[1] = CONST64(0x629a292a367cd507); + md->sha512.state[2] = CONST64(0x9159015a3070dd17); + md->sha512.state[3] = CONST64(0x152fecd8f70e5939); + md->sha512.state[4] = CONST64(0x67332667ffc00b31); + md->sha512.state[5] = CONST64(0x8eb44a8768581511); + md->sha512.state[6] = CONST64(0xdb0c2e0d64f98fa7); + md->sha512.state[7] = CONST64(0x47b5481dbefa4fa4); + return CRYPT_OK; +} + +/** + Terminate the hash to get the digest + @param md The hash state + @param out [out] The destination of the hash (48 bytes) + @return CRYPT_OK if successful +*/ +int sha384_done(hash_state * md, unsigned char *out) +{ + unsigned char buf[64]; + + LTC_ARGCHK(md != NULL); + LTC_ARGCHK(out != NULL); + + if (md->sha512.curlen >= sizeof(md->sha512.buf)) { + return CRYPT_INVALID_ARG; + } + + sha512_done(md, buf); + XMEMCPY(out, buf, 48); +#ifdef LTC_CLEAN_STACK + zeromem(buf, sizeof(buf)); +#endif + return CRYPT_OK; +} + +/** + Self-test the hash + @return CRYPT_OK if successful, CRYPT_NOP if self-tests have been disabled +*/ +int sha384_test(void) +{ + #ifndef LTC_TEST + return CRYPT_NOP; + #else + static const struct { + char *msg; + unsigned char hash[48]; + } tests[] = { + { "abc", + { 0xcb, 0x00, 0x75, 0x3f, 0x45, 0xa3, 0x5e, 0x8b, + 0xb5, 0xa0, 0x3d, 0x69, 0x9a, 0xc6, 0x50, 0x07, + 0x27, 0x2c, 0x32, 0xab, 0x0e, 0xde, 0xd1, 0x63, + 0x1a, 0x8b, 0x60, 0x5a, 0x43, 0xff, 0x5b, 0xed, + 0x80, 0x86, 0x07, 0x2b, 0xa1, 0xe7, 0xcc, 0x23, + 0x58, 0xba, 0xec, 0xa1, 0x34, 0xc8, 0x25, 0xa7 } + }, + { "abcdefghbcdefghicdefghijdefghijkefghijklfghijklmghijklmnhijklmnoijklmnopjklmnopqklmnopqrlmnopqrsmnopqrstnopqrstu", + { 0x09, 0x33, 0x0c, 0x33, 0xf7, 0x11, 0x47, 0xe8, + 0x3d, 0x19, 0x2f, 0xc7, 0x82, 0xcd, 0x1b, 0x47, + 0x53, 0x11, 0x1b, 0x17, 0x3b, 0x3b, 0x05, 0xd2, + 0x2f, 0xa0, 0x80, 0x86, 0xe3, 0xb0, 0xf7, 0x12, + 0xfc, 0xc7, 0xc7, 0x1a, 0x55, 0x7e, 0x2d, 0xb9, + 0x66, 0xc3, 0xe9, 0xfa, 0x91, 0x74, 0x60, 0x39 } + }, + }; + + int i; + unsigned char tmp[48]; + hash_state md; + + for (i = 0; i < (int)(sizeof(tests) / sizeof(tests[0])); i++) { + sha384_init(&md); + sha384_process(&md, (unsigned char*)tests[i].msg, (unsigned long)strlen(tests[i].msg)); + sha384_done(&md, tmp); + if (XMEMCMP(tmp, tests[i].hash, 48) != 0) { + return CRYPT_FAIL_TESTVECTOR; + } + } + return CRYPT_OK; + #endif +} + +#endif /* defined(LTC_SHA384) && defined(LTC_SHA512) */ + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @param sha512.c + LTC_SHA512 by Tom St Denis +*/ + +#ifdef LTC_SHA512 + +const struct ltc_hash_descriptor sha512_desc = +{ + "sha512", + 5, + 64, + 128, + + /* OID */ + { 2, 16, 840, 1, 101, 3, 4, 2, 3, }, + 9, + + &sha512_init, + &sha512_process, + &sha512_done, + &sha512_test, + NULL +}; + +/* the K array */ +static const ulong64 K[80] = { +CONST64(0x428a2f98d728ae22), CONST64(0x7137449123ef65cd), +CONST64(0xb5c0fbcfec4d3b2f), CONST64(0xe9b5dba58189dbbc), +CONST64(0x3956c25bf348b538), CONST64(0x59f111f1b605d019), +CONST64(0x923f82a4af194f9b), CONST64(0xab1c5ed5da6d8118), +CONST64(0xd807aa98a3030242), CONST64(0x12835b0145706fbe), +CONST64(0x243185be4ee4b28c), CONST64(0x550c7dc3d5ffb4e2), +CONST64(0x72be5d74f27b896f), CONST64(0x80deb1fe3b1696b1), +CONST64(0x9bdc06a725c71235), CONST64(0xc19bf174cf692694), +CONST64(0xe49b69c19ef14ad2), CONST64(0xefbe4786384f25e3), +CONST64(0x0fc19dc68b8cd5b5), CONST64(0x240ca1cc77ac9c65), +CONST64(0x2de92c6f592b0275), CONST64(0x4a7484aa6ea6e483), +CONST64(0x5cb0a9dcbd41fbd4), CONST64(0x76f988da831153b5), +CONST64(0x983e5152ee66dfab), CONST64(0xa831c66d2db43210), +CONST64(0xb00327c898fb213f), CONST64(0xbf597fc7beef0ee4), +CONST64(0xc6e00bf33da88fc2), CONST64(0xd5a79147930aa725), +CONST64(0x06ca6351e003826f), CONST64(0x142929670a0e6e70), +CONST64(0x27b70a8546d22ffc), CONST64(0x2e1b21385c26c926), +CONST64(0x4d2c6dfc5ac42aed), CONST64(0x53380d139d95b3df), +CONST64(0x650a73548baf63de), CONST64(0x766a0abb3c77b2a8), +CONST64(0x81c2c92e47edaee6), CONST64(0x92722c851482353b), +CONST64(0xa2bfe8a14cf10364), CONST64(0xa81a664bbc423001), +CONST64(0xc24b8b70d0f89791), CONST64(0xc76c51a30654be30), +CONST64(0xd192e819d6ef5218), CONST64(0xd69906245565a910), +CONST64(0xf40e35855771202a), CONST64(0x106aa07032bbd1b8), +CONST64(0x19a4c116b8d2d0c8), CONST64(0x1e376c085141ab53), +CONST64(0x2748774cdf8eeb99), CONST64(0x34b0bcb5e19b48a8), +CONST64(0x391c0cb3c5c95a63), CONST64(0x4ed8aa4ae3418acb), +CONST64(0x5b9cca4f7763e373), CONST64(0x682e6ff3d6b2b8a3), +CONST64(0x748f82ee5defb2fc), CONST64(0x78a5636f43172f60), +CONST64(0x84c87814a1f0ab72), CONST64(0x8cc702081a6439ec), +CONST64(0x90befffa23631e28), CONST64(0xa4506cebde82bde9), +CONST64(0xbef9a3f7b2c67915), CONST64(0xc67178f2e372532b), +CONST64(0xca273eceea26619c), CONST64(0xd186b8c721c0c207), +CONST64(0xeada7dd6cde0eb1e), CONST64(0xf57d4f7fee6ed178), +CONST64(0x06f067aa72176fba), CONST64(0x0a637dc5a2c898a6), +CONST64(0x113f9804bef90dae), CONST64(0x1b710b35131c471b), +CONST64(0x28db77f523047d84), CONST64(0x32caab7b40c72493), +CONST64(0x3c9ebe0a15c9bebc), CONST64(0x431d67c49c100d4c), +CONST64(0x4cc5d4becb3e42b6), CONST64(0x597f299cfc657e2a), +CONST64(0x5fcb6fab3ad6faec), CONST64(0x6c44198c4a475817) +}; + +/* Various logical functions */ +#undef S +#undef R +#undef Sigma0 +#undef Sigma1 +#undef Gamma0 +#undef Gamma1 + +#define Ch(x,y,z) (z ^ (x & (y ^ z))) +#define Maj(x,y,z) (((x | y) & z) | (x & y)) +#define S(x, n) ROR64c(x, n) +#define R(x, n) (((x)&CONST64(0xFFFFFFFFFFFFFFFF))>>((ulong64)n)) +#define Sigma0(x) (S(x, 28) ^ S(x, 34) ^ S(x, 39)) +#define Sigma1(x) (S(x, 14) ^ S(x, 18) ^ S(x, 41)) +#define Gamma0(x) (S(x, 1) ^ S(x, 8) ^ R(x, 7)) +#define Gamma1(x) (S(x, 19) ^ S(x, 61) ^ R(x, 6)) + +/* compress 1024-bits */ +#ifdef LTC_CLEAN_STACK +static int _sha512_compress(hash_state * md, unsigned char *buf) +#else +static int sha512_compress(hash_state * md, unsigned char *buf) +#endif +{ + ulong64 S[8], W[80], t0, t1; + int i; + + /* copy state into S */ + for (i = 0; i < 8; i++) { + S[i] = md->sha512.state[i]; + } + + /* copy the state into 1024-bits into W[0..15] */ + for (i = 0; i < 16; i++) { + LOAD64H(W[i], buf + (8*i)); + } + + /* fill W[16..79] */ + for (i = 16; i < 80; i++) { + W[i] = Gamma1(W[i - 2]) + W[i - 7] + Gamma0(W[i - 15]) + W[i - 16]; + } + + /* Compress */ +#ifdef LTC_SMALL_CODE + for (i = 0; i < 80; i++) { + t0 = S[7] + Sigma1(S[4]) + Ch(S[4], S[5], S[6]) + K[i] + W[i]; + t1 = Sigma0(S[0]) + Maj(S[0], S[1], S[2]); + S[7] = S[6]; + S[6] = S[5]; + S[5] = S[4]; + S[4] = S[3] + t0; + S[3] = S[2]; + S[2] = S[1]; + S[1] = S[0]; + S[0] = t0 + t1; + } +#else +#define RND(a,b,c,d,e,f,g,h,i) \ + t0 = h + Sigma1(e) + Ch(e, f, g) + K[i] + W[i]; \ + t1 = Sigma0(a) + Maj(a, b, c); \ + d += t0; \ + h = t0 + t1; + + for (i = 0; i < 80; i += 8) { + RND(S[0],S[1],S[2],S[3],S[4],S[5],S[6],S[7],i+0); + RND(S[7],S[0],S[1],S[2],S[3],S[4],S[5],S[6],i+1); + RND(S[6],S[7],S[0],S[1],S[2],S[3],S[4],S[5],i+2); + RND(S[5],S[6],S[7],S[0],S[1],S[2],S[3],S[4],i+3); + RND(S[4],S[5],S[6],S[7],S[0],S[1],S[2],S[3],i+4); + RND(S[3],S[4],S[5],S[6],S[7],S[0],S[1],S[2],i+5); + RND(S[2],S[3],S[4],S[5],S[6],S[7],S[0],S[1],i+6); + RND(S[1],S[2],S[3],S[4],S[5],S[6],S[7],S[0],i+7); + } +#endif + + + /* feedback */ + for (i = 0; i < 8; i++) { + md->sha512.state[i] = md->sha512.state[i] + S[i]; + } + + return CRYPT_OK; +} + +/* compress 1024-bits */ +#ifdef LTC_CLEAN_STACK +static int sha512_compress(hash_state * md, unsigned char *buf) +{ + int err; + err = _sha512_compress(md, buf); + burn_stack(sizeof(ulong64) * 90 + sizeof(int)); + return err; +} +#endif + +/** + Initialize the hash state + @param md The hash state you wish to initialize + @return CRYPT_OK if successful +*/ +int sha512_init(hash_state * md) +{ + LTC_ARGCHK(md != NULL); + md->sha512.curlen = 0; + md->sha512.length = 0; + md->sha512.state[0] = CONST64(0x6a09e667f3bcc908); + md->sha512.state[1] = CONST64(0xbb67ae8584caa73b); + md->sha512.state[2] = CONST64(0x3c6ef372fe94f82b); + md->sha512.state[3] = CONST64(0xa54ff53a5f1d36f1); + md->sha512.state[4] = CONST64(0x510e527fade682d1); + md->sha512.state[5] = CONST64(0x9b05688c2b3e6c1f); + md->sha512.state[6] = CONST64(0x1f83d9abfb41bd6b); + md->sha512.state[7] = CONST64(0x5be0cd19137e2179); + return CRYPT_OK; +} + +/** + Process a block of memory though the hash + @param md The hash state + @param in The data to hash + @param inlen The length of the data (octets) + @return CRYPT_OK if successful +*/ +HASH_PROCESS(sha512_process, sha512_compress, sha512, 128) + +/** + Terminate the hash to get the digest + @param md The hash state + @param out [out] The destination of the hash (64 bytes) + @return CRYPT_OK if successful +*/ +int sha512_done(hash_state * md, unsigned char *out) +{ + int i; + + LTC_ARGCHK(md != NULL); + LTC_ARGCHK(out != NULL); + + if (md->sha512.curlen >= sizeof(md->sha512.buf)) { + return CRYPT_INVALID_ARG; + } + + /* increase the length of the message */ + md->sha512.length += md->sha512.curlen * CONST64(8); + + /* append the '1' bit */ + md->sha512.buf[md->sha512.curlen++] = (unsigned char)0x80; + + /* if the length is currently above 112 bytes we append zeros + * then compress. Then we can fall back to padding zeros and length + * encoding like normal. + */ + if (md->sha512.curlen > 112) { + while (md->sha512.curlen < 128) { + md->sha512.buf[md->sha512.curlen++] = (unsigned char)0; + } + sha512_compress(md, md->sha512.buf); + md->sha512.curlen = 0; + } + + /* pad upto 120 bytes of zeroes + * note: that from 112 to 120 is the 64 MSB of the length. We assume that you won't hash + * > 2^64 bits of data... :-) + */ + while (md->sha512.curlen < 120) { + md->sha512.buf[md->sha512.curlen++] = (unsigned char)0; + } + + /* store length */ + STORE64H(md->sha512.length, md->sha512.buf+120); + sha512_compress(md, md->sha512.buf); + + /* copy output */ + for (i = 0; i < 8; i++) { + STORE64H(md->sha512.state[i], out+(8*i)); + } +#ifdef LTC_CLEAN_STACK + zeromem(md, sizeof(hash_state)); +#endif + return CRYPT_OK; +} + +/** + Self-test the hash + @return CRYPT_OK if successful, CRYPT_NOP if self-tests have been disabled +*/ +int sha512_test(void) +{ + #ifndef LTC_TEST + return CRYPT_NOP; + #else + static const struct { + char *msg; + unsigned char hash[64]; + } tests[] = { + { "abc", + { 0xdd, 0xaf, 0x35, 0xa1, 0x93, 0x61, 0x7a, 0xba, + 0xcc, 0x41, 0x73, 0x49, 0xae, 0x20, 0x41, 0x31, + 0x12, 0xe6, 0xfa, 0x4e, 0x89, 0xa9, 0x7e, 0xa2, + 0x0a, 0x9e, 0xee, 0xe6, 0x4b, 0x55, 0xd3, 0x9a, + 0x21, 0x92, 0x99, 0x2a, 0x27, 0x4f, 0xc1, 0xa8, + 0x36, 0xba, 0x3c, 0x23, 0xa3, 0xfe, 0xeb, 0xbd, + 0x45, 0x4d, 0x44, 0x23, 0x64, 0x3c, 0xe8, 0x0e, + 0x2a, 0x9a, 0xc9, 0x4f, 0xa5, 0x4c, 0xa4, 0x9f } + }, + { "abcdefghbcdefghicdefghijdefghijkefghijklfghijklmghijklmnhijklmnoijklmnopjklmnopqklmnopqrlmnopqrsmnopqrstnopqrstu", + { 0x8e, 0x95, 0x9b, 0x75, 0xda, 0xe3, 0x13, 0xda, + 0x8c, 0xf4, 0xf7, 0x28, 0x14, 0xfc, 0x14, 0x3f, + 0x8f, 0x77, 0x79, 0xc6, 0xeb, 0x9f, 0x7f, 0xa1, + 0x72, 0x99, 0xae, 0xad, 0xb6, 0x88, 0x90, 0x18, + 0x50, 0x1d, 0x28, 0x9e, 0x49, 0x00, 0xf7, 0xe4, + 0x33, 0x1b, 0x99, 0xde, 0xc4, 0xb5, 0x43, 0x3a, + 0xc7, 0xd3, 0x29, 0xee, 0xb6, 0xdd, 0x26, 0x54, + 0x5e, 0x96, 0xe5, 0x5b, 0x87, 0x4b, 0xe9, 0x09 } + }, + }; + + int i; + unsigned char tmp[64]; + hash_state md; + + for (i = 0; i < (int)(sizeof(tests) / sizeof(tests[0])); i++) { + sha512_init(&md); + sha512_process(&md, (unsigned char *)tests[i].msg, (unsigned long)strlen(tests[i].msg)); + sha512_done(&md, tmp); + if (XMEMCMP(tmp, tests[i].hash, 64) != 0) { + return CRYPT_FAIL_TESTVECTOR; + } + } + return CRYPT_OK; + #endif +} + +#endif + + + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file hmac_init.c + HMAC support, initialize state, Tom St Denis/Dobes Vandermeer +*/ + +#ifdef LTC_HMAC + +#define LTC_HMAC_BLOCKSIZE hash_descriptor[hash].blocksize + +/** + Initialize an HMAC context. + @param hmac The HMAC state + @param hash The index of the hash you want to use + @param key The secret key + @param keylen The length of the secret key (octets) + @return CRYPT_OK if successful +*/ +int hmac_init(hmac_state *hmac, int hash, const unsigned char *key, unsigned long keylen) +{ + unsigned char *buf; + unsigned long hashsize; + unsigned long i, z; + int err; + + LTC_ARGCHK(hmac != NULL); + LTC_ARGCHK(key != NULL); + + /* valid hash? */ + if ((err = hash_is_valid(hash)) != CRYPT_OK) { + return err; + } + hmac->hash = hash; + hashsize = hash_descriptor[hash].hashsize; + + /* valid key length? */ + if (keylen == 0) { + return CRYPT_INVALID_KEYSIZE; + } + + /* allocate ram for buf */ + buf = XMALLOC(LTC_HMAC_BLOCKSIZE); + if (buf == NULL) { + return CRYPT_MEM; + } + + /* allocate memory for key */ + hmac->key = XMALLOC(LTC_HMAC_BLOCKSIZE); + if (hmac->key == NULL) { + XFREE(buf); + return CRYPT_MEM; + } + + /* (1) make sure we have a large enough key */ + if(keylen > LTC_HMAC_BLOCKSIZE) { + z = LTC_HMAC_BLOCKSIZE; + if ((err = hash_memory(hash, key, keylen, hmac->key, &z)) != CRYPT_OK) { + goto LBL_ERR; + } + keylen = hashsize; + } else { + XMEMCPY(hmac->key, key, (size_t)keylen); + } + + if(keylen < LTC_HMAC_BLOCKSIZE) { + zeromem((hmac->key) + keylen, (size_t)(LTC_HMAC_BLOCKSIZE - keylen)); + } + + /* Create the initial vector for step (3) */ + for(i=0; i < LTC_HMAC_BLOCKSIZE; i++) { + buf[i] = hmac->key[i] ^ 0x36; + } + + /* Pre-pend that to the hash data */ + if ((err = hash_descriptor[hash].init(&hmac->md)) != CRYPT_OK) { + goto LBL_ERR; + } + + if ((err = hash_descriptor[hash].process(&hmac->md, buf, LTC_HMAC_BLOCKSIZE)) != CRYPT_OK) { + goto LBL_ERR; + } + goto done; +LBL_ERR: + /* free the key since we failed */ + XFREE(hmac->key); +done: +#ifdef LTC_CLEAN_STACK + zeromem(buf, LTC_HMAC_BLOCKSIZE); +#endif + + XFREE(buf); + return err; +} + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file hmac_process.c + HMAC support, process data, Tom St Denis/Dobes Vandermeer +*/ + +#ifdef LTC_HMAC + +/** + Process data through HMAC + @param hmac The hmac state + @param in The data to send through HMAC + @param inlen The length of the data to HMAC (octets) + @return CRYPT_OK if successful +*/ +int hmac_process(hmac_state *hmac, const unsigned char *in, unsigned long inlen) +{ + int err; + LTC_ARGCHK(hmac != NULL); + LTC_ARGCHK(in != NULL); + if ((err = hash_is_valid(hmac->hash)) != CRYPT_OK) { + return err; + } + return hash_descriptor[hmac->hash].process(&hmac->md, in, inlen); +} + +#endif + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file hmac_done.c + HMAC support, terminate stream, Tom St Denis/Dobes Vandermeer +*/ + +#ifdef LTC_HMAC + +#define LTC_HMAC_BLOCKSIZE hash_descriptor[hash].blocksize + +/** + Terminate an HMAC session + @param hmac The HMAC state + @param out [out] The destination of the HMAC authentication tag + @param outlen [in/out] The max size and resulting size of the HMAC authentication tag + @return CRYPT_OK if successful +*/ +int hmac_done(hmac_state *hmac, unsigned char *out, unsigned long *outlen) +{ + unsigned char *buf, *isha; + unsigned long hashsize, i; + int hash, err; + + LTC_ARGCHK(hmac != NULL); + LTC_ARGCHK(out != NULL); + + /* test hash */ + hash = hmac->hash; + if((err = hash_is_valid(hash)) != CRYPT_OK) { + return err; + } + + /* get the hash message digest size */ + hashsize = hash_descriptor[hash].hashsize; + + /* allocate buffers */ + buf = XMALLOC(LTC_HMAC_BLOCKSIZE); + isha = XMALLOC(hashsize); + if (buf == NULL || isha == NULL) { + if (buf != NULL) { + XFREE(buf); + } + if (isha != NULL) { + XFREE(isha); + } + return CRYPT_MEM; + } + + /* Get the hash of the first HMAC vector plus the data */ + if ((err = hash_descriptor[hash].done(&hmac->md, isha)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* Create the second HMAC vector vector for step (3) */ + for(i=0; i < LTC_HMAC_BLOCKSIZE; i++) { + buf[i] = hmac->key[i] ^ 0x5C; + } + + /* Now calculate the "outer" hash for step (5), (6), and (7) */ + if ((err = hash_descriptor[hash].init(&hmac->md)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash].process(&hmac->md, buf, LTC_HMAC_BLOCKSIZE)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash].process(&hmac->md, isha, hashsize)) != CRYPT_OK) { + goto LBL_ERR; + } + if ((err = hash_descriptor[hash].done(&hmac->md, buf)) != CRYPT_OK) { + goto LBL_ERR; + } + + /* copy to output */ + for (i = 0; i < hashsize && i < *outlen; i++) { + out[i] = buf[i]; + } + *outlen = i; + + err = CRYPT_OK; +LBL_ERR: + XFREE(hmac->key); +#ifdef LTC_CLEAN_STACK + zeromem(isha, hashsize); + zeromem(buf, hashsize); + zeromem(hmac, sizeof(*hmac)); +#endif + + XFREE(isha); + XFREE(buf); + + return err; +} + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +#define __LTC_AES_TAB_C__ +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ +/* The precomputed tables for AES */ +/* +Te0[x] = S [x].[02, 01, 01, 03]; +Te1[x] = S [x].[03, 02, 01, 01]; +Te2[x] = S [x].[01, 03, 02, 01]; +Te3[x] = S [x].[01, 01, 03, 02]; +Te4[x] = S [x].[01, 01, 01, 01]; + +Td0[x] = Si[x].[0e, 09, 0d, 0b]; +Td1[x] = Si[x].[0b, 0e, 09, 0d]; +Td2[x] = Si[x].[0d, 0b, 0e, 09]; +Td3[x] = Si[x].[09, 0d, 0b, 0e]; +Td4[x] = Si[x].[01, 01, 01, 01]; +*/ + +#ifdef __LTC_AES_TAB_C__ + +/** + @file aes_tab.c + AES tables +*/ +static const ulong32 TE0[256] = { + 0xc66363a5UL, 0xf87c7c84UL, 0xee777799UL, 0xf67b7b8dUL, + 0xfff2f20dUL, 0xd66b6bbdUL, 0xde6f6fb1UL, 0x91c5c554UL, + 0x60303050UL, 0x02010103UL, 0xce6767a9UL, 0x562b2b7dUL, + 0xe7fefe19UL, 0xb5d7d762UL, 0x4dababe6UL, 0xec76769aUL, + 0x8fcaca45UL, 0x1f82829dUL, 0x89c9c940UL, 0xfa7d7d87UL, + 0xeffafa15UL, 0xb25959ebUL, 0x8e4747c9UL, 0xfbf0f00bUL, + 0x41adadecUL, 0xb3d4d467UL, 0x5fa2a2fdUL, 0x45afafeaUL, + 0x239c9cbfUL, 0x53a4a4f7UL, 0xe4727296UL, 0x9bc0c05bUL, + 0x75b7b7c2UL, 0xe1fdfd1cUL, 0x3d9393aeUL, 0x4c26266aUL, + 0x6c36365aUL, 0x7e3f3f41UL, 0xf5f7f702UL, 0x83cccc4fUL, + 0x6834345cUL, 0x51a5a5f4UL, 0xd1e5e534UL, 0xf9f1f108UL, + 0xe2717193UL, 0xabd8d873UL, 0x62313153UL, 0x2a15153fUL, + 0x0804040cUL, 0x95c7c752UL, 0x46232365UL, 0x9dc3c35eUL, + 0x30181828UL, 0x379696a1UL, 0x0a05050fUL, 0x2f9a9ab5UL, + 0x0e070709UL, 0x24121236UL, 0x1b80809bUL, 0xdfe2e23dUL, + 0xcdebeb26UL, 0x4e272769UL, 0x7fb2b2cdUL, 0xea75759fUL, + 0x1209091bUL, 0x1d83839eUL, 0x582c2c74UL, 0x341a1a2eUL, + 0x361b1b2dUL, 0xdc6e6eb2UL, 0xb45a5aeeUL, 0x5ba0a0fbUL, + 0xa45252f6UL, 0x763b3b4dUL, 0xb7d6d661UL, 0x7db3b3ceUL, + 0x5229297bUL, 0xdde3e33eUL, 0x5e2f2f71UL, 0x13848497UL, + 0xa65353f5UL, 0xb9d1d168UL, 0x00000000UL, 0xc1eded2cUL, + 0x40202060UL, 0xe3fcfc1fUL, 0x79b1b1c8UL, 0xb65b5bedUL, + 0xd46a6abeUL, 0x8dcbcb46UL, 0x67bebed9UL, 0x7239394bUL, + 0x944a4adeUL, 0x984c4cd4UL, 0xb05858e8UL, 0x85cfcf4aUL, + 0xbbd0d06bUL, 0xc5efef2aUL, 0x4faaaae5UL, 0xedfbfb16UL, + 0x864343c5UL, 0x9a4d4dd7UL, 0x66333355UL, 0x11858594UL, + 0x8a4545cfUL, 0xe9f9f910UL, 0x04020206UL, 0xfe7f7f81UL, + 0xa05050f0UL, 0x783c3c44UL, 0x259f9fbaUL, 0x4ba8a8e3UL, + 0xa25151f3UL, 0x5da3a3feUL, 0x804040c0UL, 0x058f8f8aUL, + 0x3f9292adUL, 0x219d9dbcUL, 0x70383848UL, 0xf1f5f504UL, + 0x63bcbcdfUL, 0x77b6b6c1UL, 0xafdada75UL, 0x42212163UL, + 0x20101030UL, 0xe5ffff1aUL, 0xfdf3f30eUL, 0xbfd2d26dUL, + 0x81cdcd4cUL, 0x180c0c14UL, 0x26131335UL, 0xc3ecec2fUL, + 0xbe5f5fe1UL, 0x359797a2UL, 0x884444ccUL, 0x2e171739UL, + 0x93c4c457UL, 0x55a7a7f2UL, 0xfc7e7e82UL, 0x7a3d3d47UL, + 0xc86464acUL, 0xba5d5de7UL, 0x3219192bUL, 0xe6737395UL, + 0xc06060a0UL, 0x19818198UL, 0x9e4f4fd1UL, 0xa3dcdc7fUL, + 0x44222266UL, 0x542a2a7eUL, 0x3b9090abUL, 0x0b888883UL, + 0x8c4646caUL, 0xc7eeee29UL, 0x6bb8b8d3UL, 0x2814143cUL, + 0xa7dede79UL, 0xbc5e5ee2UL, 0x160b0b1dUL, 0xaddbdb76UL, + 0xdbe0e03bUL, 0x64323256UL, 0x743a3a4eUL, 0x140a0a1eUL, + 0x924949dbUL, 0x0c06060aUL, 0x4824246cUL, 0xb85c5ce4UL, + 0x9fc2c25dUL, 0xbdd3d36eUL, 0x43acacefUL, 0xc46262a6UL, + 0x399191a8UL, 0x319595a4UL, 0xd3e4e437UL, 0xf279798bUL, + 0xd5e7e732UL, 0x8bc8c843UL, 0x6e373759UL, 0xda6d6db7UL, + 0x018d8d8cUL, 0xb1d5d564UL, 0x9c4e4ed2UL, 0x49a9a9e0UL, + 0xd86c6cb4UL, 0xac5656faUL, 0xf3f4f407UL, 0xcfeaea25UL, + 0xca6565afUL, 0xf47a7a8eUL, 0x47aeaee9UL, 0x10080818UL, + 0x6fbabad5UL, 0xf0787888UL, 0x4a25256fUL, 0x5c2e2e72UL, + 0x381c1c24UL, 0x57a6a6f1UL, 0x73b4b4c7UL, 0x97c6c651UL, + 0xcbe8e823UL, 0xa1dddd7cUL, 0xe874749cUL, 0x3e1f1f21UL, + 0x964b4bddUL, 0x61bdbddcUL, 0x0d8b8b86UL, 0x0f8a8a85UL, + 0xe0707090UL, 0x7c3e3e42UL, 0x71b5b5c4UL, 0xcc6666aaUL, + 0x904848d8UL, 0x06030305UL, 0xf7f6f601UL, 0x1c0e0e12UL, + 0xc26161a3UL, 0x6a35355fUL, 0xae5757f9UL, 0x69b9b9d0UL, + 0x17868691UL, 0x99c1c158UL, 0x3a1d1d27UL, 0x279e9eb9UL, + 0xd9e1e138UL, 0xebf8f813UL, 0x2b9898b3UL, 0x22111133UL, + 0xd26969bbUL, 0xa9d9d970UL, 0x078e8e89UL, 0x339494a7UL, + 0x2d9b9bb6UL, 0x3c1e1e22UL, 0x15878792UL, 0xc9e9e920UL, + 0x87cece49UL, 0xaa5555ffUL, 0x50282878UL, 0xa5dfdf7aUL, + 0x038c8c8fUL, 0x59a1a1f8UL, 0x09898980UL, 0x1a0d0d17UL, + 0x65bfbfdaUL, 0xd7e6e631UL, 0x844242c6UL, 0xd06868b8UL, + 0x824141c3UL, 0x299999b0UL, 0x5a2d2d77UL, 0x1e0f0f11UL, + 0x7bb0b0cbUL, 0xa85454fcUL, 0x6dbbbbd6UL, 0x2c16163aUL, +}; + +#ifndef PELI_TAB +static const ulong32 Te4[256] = { + 0x63636363UL, 0x7c7c7c7cUL, 0x77777777UL, 0x7b7b7b7bUL, + 0xf2f2f2f2UL, 0x6b6b6b6bUL, 0x6f6f6f6fUL, 0xc5c5c5c5UL, + 0x30303030UL, 0x01010101UL, 0x67676767UL, 0x2b2b2b2bUL, + 0xfefefefeUL, 0xd7d7d7d7UL, 0xababababUL, 0x76767676UL, + 0xcacacacaUL, 0x82828282UL, 0xc9c9c9c9UL, 0x7d7d7d7dUL, + 0xfafafafaUL, 0x59595959UL, 0x47474747UL, 0xf0f0f0f0UL, + 0xadadadadUL, 0xd4d4d4d4UL, 0xa2a2a2a2UL, 0xafafafafUL, + 0x9c9c9c9cUL, 0xa4a4a4a4UL, 0x72727272UL, 0xc0c0c0c0UL, + 0xb7b7b7b7UL, 0xfdfdfdfdUL, 0x93939393UL, 0x26262626UL, + 0x36363636UL, 0x3f3f3f3fUL, 0xf7f7f7f7UL, 0xccccccccUL, + 0x34343434UL, 0xa5a5a5a5UL, 0xe5e5e5e5UL, 0xf1f1f1f1UL, + 0x71717171UL, 0xd8d8d8d8UL, 0x31313131UL, 0x15151515UL, + 0x04040404UL, 0xc7c7c7c7UL, 0x23232323UL, 0xc3c3c3c3UL, + 0x18181818UL, 0x96969696UL, 0x05050505UL, 0x9a9a9a9aUL, + 0x07070707UL, 0x12121212UL, 0x80808080UL, 0xe2e2e2e2UL, + 0xebebebebUL, 0x27272727UL, 0xb2b2b2b2UL, 0x75757575UL, + 0x09090909UL, 0x83838383UL, 0x2c2c2c2cUL, 0x1a1a1a1aUL, + 0x1b1b1b1bUL, 0x6e6e6e6eUL, 0x5a5a5a5aUL, 0xa0a0a0a0UL, + 0x52525252UL, 0x3b3b3b3bUL, 0xd6d6d6d6UL, 0xb3b3b3b3UL, + 0x29292929UL, 0xe3e3e3e3UL, 0x2f2f2f2fUL, 0x84848484UL, + 0x53535353UL, 0xd1d1d1d1UL, 0x00000000UL, 0xededededUL, + 0x20202020UL, 0xfcfcfcfcUL, 0xb1b1b1b1UL, 0x5b5b5b5bUL, + 0x6a6a6a6aUL, 0xcbcbcbcbUL, 0xbebebebeUL, 0x39393939UL, + 0x4a4a4a4aUL, 0x4c4c4c4cUL, 0x58585858UL, 0xcfcfcfcfUL, + 0xd0d0d0d0UL, 0xefefefefUL, 0xaaaaaaaaUL, 0xfbfbfbfbUL, + 0x43434343UL, 0x4d4d4d4dUL, 0x33333333UL, 0x85858585UL, + 0x45454545UL, 0xf9f9f9f9UL, 0x02020202UL, 0x7f7f7f7fUL, + 0x50505050UL, 0x3c3c3c3cUL, 0x9f9f9f9fUL, 0xa8a8a8a8UL, + 0x51515151UL, 0xa3a3a3a3UL, 0x40404040UL, 0x8f8f8f8fUL, + 0x92929292UL, 0x9d9d9d9dUL, 0x38383838UL, 0xf5f5f5f5UL, + 0xbcbcbcbcUL, 0xb6b6b6b6UL, 0xdadadadaUL, 0x21212121UL, + 0x10101010UL, 0xffffffffUL, 0xf3f3f3f3UL, 0xd2d2d2d2UL, + 0xcdcdcdcdUL, 0x0c0c0c0cUL, 0x13131313UL, 0xececececUL, + 0x5f5f5f5fUL, 0x97979797UL, 0x44444444UL, 0x17171717UL, + 0xc4c4c4c4UL, 0xa7a7a7a7UL, 0x7e7e7e7eUL, 0x3d3d3d3dUL, + 0x64646464UL, 0x5d5d5d5dUL, 0x19191919UL, 0x73737373UL, + 0x60606060UL, 0x81818181UL, 0x4f4f4f4fUL, 0xdcdcdcdcUL, + 0x22222222UL, 0x2a2a2a2aUL, 0x90909090UL, 0x88888888UL, + 0x46464646UL, 0xeeeeeeeeUL, 0xb8b8b8b8UL, 0x14141414UL, + 0xdedededeUL, 0x5e5e5e5eUL, 0x0b0b0b0bUL, 0xdbdbdbdbUL, + 0xe0e0e0e0UL, 0x32323232UL, 0x3a3a3a3aUL, 0x0a0a0a0aUL, + 0x49494949UL, 0x06060606UL, 0x24242424UL, 0x5c5c5c5cUL, + 0xc2c2c2c2UL, 0xd3d3d3d3UL, 0xacacacacUL, 0x62626262UL, + 0x91919191UL, 0x95959595UL, 0xe4e4e4e4UL, 0x79797979UL, + 0xe7e7e7e7UL, 0xc8c8c8c8UL, 0x37373737UL, 0x6d6d6d6dUL, + 0x8d8d8d8dUL, 0xd5d5d5d5UL, 0x4e4e4e4eUL, 0xa9a9a9a9UL, + 0x6c6c6c6cUL, 0x56565656UL, 0xf4f4f4f4UL, 0xeaeaeaeaUL, + 0x65656565UL, 0x7a7a7a7aUL, 0xaeaeaeaeUL, 0x08080808UL, + 0xbabababaUL, 0x78787878UL, 0x25252525UL, 0x2e2e2e2eUL, + 0x1c1c1c1cUL, 0xa6a6a6a6UL, 0xb4b4b4b4UL, 0xc6c6c6c6UL, + 0xe8e8e8e8UL, 0xddddddddUL, 0x74747474UL, 0x1f1f1f1fUL, + 0x4b4b4b4bUL, 0xbdbdbdbdUL, 0x8b8b8b8bUL, 0x8a8a8a8aUL, + 0x70707070UL, 0x3e3e3e3eUL, 0xb5b5b5b5UL, 0x66666666UL, + 0x48484848UL, 0x03030303UL, 0xf6f6f6f6UL, 0x0e0e0e0eUL, + 0x61616161UL, 0x35353535UL, 0x57575757UL, 0xb9b9b9b9UL, + 0x86868686UL, 0xc1c1c1c1UL, 0x1d1d1d1dUL, 0x9e9e9e9eUL, + 0xe1e1e1e1UL, 0xf8f8f8f8UL, 0x98989898UL, 0x11111111UL, + 0x69696969UL, 0xd9d9d9d9UL, 0x8e8e8e8eUL, 0x94949494UL, + 0x9b9b9b9bUL, 0x1e1e1e1eUL, 0x87878787UL, 0xe9e9e9e9UL, + 0xcecececeUL, 0x55555555UL, 0x28282828UL, 0xdfdfdfdfUL, + 0x8c8c8c8cUL, 0xa1a1a1a1UL, 0x89898989UL, 0x0d0d0d0dUL, + 0xbfbfbfbfUL, 0xe6e6e6e6UL, 0x42424242UL, 0x68686868UL, + 0x41414141UL, 0x99999999UL, 0x2d2d2d2dUL, 0x0f0f0f0fUL, + 0xb0b0b0b0UL, 0x54545454UL, 0xbbbbbbbbUL, 0x16161616UL, +}; +#endif + +#ifndef ENCRYPT_ONLY + +static const ulong32 TD0[256] = { + 0x51f4a750UL, 0x7e416553UL, 0x1a17a4c3UL, 0x3a275e96UL, + 0x3bab6bcbUL, 0x1f9d45f1UL, 0xacfa58abUL, 0x4be30393UL, + 0x2030fa55UL, 0xad766df6UL, 0x88cc7691UL, 0xf5024c25UL, + 0x4fe5d7fcUL, 0xc52acbd7UL, 0x26354480UL, 0xb562a38fUL, + 0xdeb15a49UL, 0x25ba1b67UL, 0x45ea0e98UL, 0x5dfec0e1UL, + 0xc32f7502UL, 0x814cf012UL, 0x8d4697a3UL, 0x6bd3f9c6UL, + 0x038f5fe7UL, 0x15929c95UL, 0xbf6d7aebUL, 0x955259daUL, + 0xd4be832dUL, 0x587421d3UL, 0x49e06929UL, 0x8ec9c844UL, + 0x75c2896aUL, 0xf48e7978UL, 0x99583e6bUL, 0x27b971ddUL, + 0xbee14fb6UL, 0xf088ad17UL, 0xc920ac66UL, 0x7dce3ab4UL, + 0x63df4a18UL, 0xe51a3182UL, 0x97513360UL, 0x62537f45UL, + 0xb16477e0UL, 0xbb6bae84UL, 0xfe81a01cUL, 0xf9082b94UL, + 0x70486858UL, 0x8f45fd19UL, 0x94de6c87UL, 0x527bf8b7UL, + 0xab73d323UL, 0x724b02e2UL, 0xe31f8f57UL, 0x6655ab2aUL, + 0xb2eb2807UL, 0x2fb5c203UL, 0x86c57b9aUL, 0xd33708a5UL, + 0x302887f2UL, 0x23bfa5b2UL, 0x02036abaUL, 0xed16825cUL, + 0x8acf1c2bUL, 0xa779b492UL, 0xf307f2f0UL, 0x4e69e2a1UL, + 0x65daf4cdUL, 0x0605bed5UL, 0xd134621fUL, 0xc4a6fe8aUL, + 0x342e539dUL, 0xa2f355a0UL, 0x058ae132UL, 0xa4f6eb75UL, + 0x0b83ec39UL, 0x4060efaaUL, 0x5e719f06UL, 0xbd6e1051UL, + 0x3e218af9UL, 0x96dd063dUL, 0xdd3e05aeUL, 0x4de6bd46UL, + 0x91548db5UL, 0x71c45d05UL, 0x0406d46fUL, 0x605015ffUL, + 0x1998fb24UL, 0xd6bde997UL, 0x894043ccUL, 0x67d99e77UL, + 0xb0e842bdUL, 0x07898b88UL, 0xe7195b38UL, 0x79c8eedbUL, + 0xa17c0a47UL, 0x7c420fe9UL, 0xf8841ec9UL, 0x00000000UL, + 0x09808683UL, 0x322bed48UL, 0x1e1170acUL, 0x6c5a724eUL, + 0xfd0efffbUL, 0x0f853856UL, 0x3daed51eUL, 0x362d3927UL, + 0x0a0fd964UL, 0x685ca621UL, 0x9b5b54d1UL, 0x24362e3aUL, + 0x0c0a67b1UL, 0x9357e70fUL, 0xb4ee96d2UL, 0x1b9b919eUL, + 0x80c0c54fUL, 0x61dc20a2UL, 0x5a774b69UL, 0x1c121a16UL, + 0xe293ba0aUL, 0xc0a02ae5UL, 0x3c22e043UL, 0x121b171dUL, + 0x0e090d0bUL, 0xf28bc7adUL, 0x2db6a8b9UL, 0x141ea9c8UL, + 0x57f11985UL, 0xaf75074cUL, 0xee99ddbbUL, 0xa37f60fdUL, + 0xf701269fUL, 0x5c72f5bcUL, 0x44663bc5UL, 0x5bfb7e34UL, + 0x8b432976UL, 0xcb23c6dcUL, 0xb6edfc68UL, 0xb8e4f163UL, + 0xd731dccaUL, 0x42638510UL, 0x13972240UL, 0x84c61120UL, + 0x854a247dUL, 0xd2bb3df8UL, 0xaef93211UL, 0xc729a16dUL, + 0x1d9e2f4bUL, 0xdcb230f3UL, 0x0d8652ecUL, 0x77c1e3d0UL, + 0x2bb3166cUL, 0xa970b999UL, 0x119448faUL, 0x47e96422UL, + 0xa8fc8cc4UL, 0xa0f03f1aUL, 0x567d2cd8UL, 0x223390efUL, + 0x87494ec7UL, 0xd938d1c1UL, 0x8ccaa2feUL, 0x98d40b36UL, + 0xa6f581cfUL, 0xa57ade28UL, 0xdab78e26UL, 0x3fadbfa4UL, + 0x2c3a9de4UL, 0x5078920dUL, 0x6a5fcc9bUL, 0x547e4662UL, + 0xf68d13c2UL, 0x90d8b8e8UL, 0x2e39f75eUL, 0x82c3aff5UL, + 0x9f5d80beUL, 0x69d0937cUL, 0x6fd52da9UL, 0xcf2512b3UL, + 0xc8ac993bUL, 0x10187da7UL, 0xe89c636eUL, 0xdb3bbb7bUL, + 0xcd267809UL, 0x6e5918f4UL, 0xec9ab701UL, 0x834f9aa8UL, + 0xe6956e65UL, 0xaaffe67eUL, 0x21bccf08UL, 0xef15e8e6UL, + 0xbae79bd9UL, 0x4a6f36ceUL, 0xea9f09d4UL, 0x29b07cd6UL, + 0x31a4b2afUL, 0x2a3f2331UL, 0xc6a59430UL, 0x35a266c0UL, + 0x744ebc37UL, 0xfc82caa6UL, 0xe090d0b0UL, 0x33a7d815UL, + 0xf104984aUL, 0x41ecdaf7UL, 0x7fcd500eUL, 0x1791f62fUL, + 0x764dd68dUL, 0x43efb04dUL, 0xccaa4d54UL, 0xe49604dfUL, + 0x9ed1b5e3UL, 0x4c6a881bUL, 0xc12c1fb8UL, 0x4665517fUL, + 0x9d5eea04UL, 0x018c355dUL, 0xfa877473UL, 0xfb0b412eUL, + 0xb3671d5aUL, 0x92dbd252UL, 0xe9105633UL, 0x6dd64713UL, + 0x9ad7618cUL, 0x37a10c7aUL, 0x59f8148eUL, 0xeb133c89UL, + 0xcea927eeUL, 0xb761c935UL, 0xe11ce5edUL, 0x7a47b13cUL, + 0x9cd2df59UL, 0x55f2733fUL, 0x1814ce79UL, 0x73c737bfUL, + 0x53f7cdeaUL, 0x5ffdaa5bUL, 0xdf3d6f14UL, 0x7844db86UL, + 0xcaaff381UL, 0xb968c43eUL, 0x3824342cUL, 0xc2a3405fUL, + 0x161dc372UL, 0xbce2250cUL, 0x283c498bUL, 0xff0d9541UL, + 0x39a80171UL, 0x080cb3deUL, 0xd8b4e49cUL, 0x6456c190UL, + 0x7bcb8461UL, 0xd532b670UL, 0x486c5c74UL, 0xd0b85742UL, +}; + +static const ulong32 Td4[256] = { + 0x52525252UL, 0x09090909UL, 0x6a6a6a6aUL, 0xd5d5d5d5UL, + 0x30303030UL, 0x36363636UL, 0xa5a5a5a5UL, 0x38383838UL, + 0xbfbfbfbfUL, 0x40404040UL, 0xa3a3a3a3UL, 0x9e9e9e9eUL, + 0x81818181UL, 0xf3f3f3f3UL, 0xd7d7d7d7UL, 0xfbfbfbfbUL, + 0x7c7c7c7cUL, 0xe3e3e3e3UL, 0x39393939UL, 0x82828282UL, + 0x9b9b9b9bUL, 0x2f2f2f2fUL, 0xffffffffUL, 0x87878787UL, + 0x34343434UL, 0x8e8e8e8eUL, 0x43434343UL, 0x44444444UL, + 0xc4c4c4c4UL, 0xdedededeUL, 0xe9e9e9e9UL, 0xcbcbcbcbUL, + 0x54545454UL, 0x7b7b7b7bUL, 0x94949494UL, 0x32323232UL, + 0xa6a6a6a6UL, 0xc2c2c2c2UL, 0x23232323UL, 0x3d3d3d3dUL, + 0xeeeeeeeeUL, 0x4c4c4c4cUL, 0x95959595UL, 0x0b0b0b0bUL, + 0x42424242UL, 0xfafafafaUL, 0xc3c3c3c3UL, 0x4e4e4e4eUL, + 0x08080808UL, 0x2e2e2e2eUL, 0xa1a1a1a1UL, 0x66666666UL, + 0x28282828UL, 0xd9d9d9d9UL, 0x24242424UL, 0xb2b2b2b2UL, + 0x76767676UL, 0x5b5b5b5bUL, 0xa2a2a2a2UL, 0x49494949UL, + 0x6d6d6d6dUL, 0x8b8b8b8bUL, 0xd1d1d1d1UL, 0x25252525UL, + 0x72727272UL, 0xf8f8f8f8UL, 0xf6f6f6f6UL, 0x64646464UL, + 0x86868686UL, 0x68686868UL, 0x98989898UL, 0x16161616UL, + 0xd4d4d4d4UL, 0xa4a4a4a4UL, 0x5c5c5c5cUL, 0xccccccccUL, + 0x5d5d5d5dUL, 0x65656565UL, 0xb6b6b6b6UL, 0x92929292UL, + 0x6c6c6c6cUL, 0x70707070UL, 0x48484848UL, 0x50505050UL, + 0xfdfdfdfdUL, 0xededededUL, 0xb9b9b9b9UL, 0xdadadadaUL, + 0x5e5e5e5eUL, 0x15151515UL, 0x46464646UL, 0x57575757UL, + 0xa7a7a7a7UL, 0x8d8d8d8dUL, 0x9d9d9d9dUL, 0x84848484UL, + 0x90909090UL, 0xd8d8d8d8UL, 0xababababUL, 0x00000000UL, + 0x8c8c8c8cUL, 0xbcbcbcbcUL, 0xd3d3d3d3UL, 0x0a0a0a0aUL, + 0xf7f7f7f7UL, 0xe4e4e4e4UL, 0x58585858UL, 0x05050505UL, + 0xb8b8b8b8UL, 0xb3b3b3b3UL, 0x45454545UL, 0x06060606UL, + 0xd0d0d0d0UL, 0x2c2c2c2cUL, 0x1e1e1e1eUL, 0x8f8f8f8fUL, + 0xcacacacaUL, 0x3f3f3f3fUL, 0x0f0f0f0fUL, 0x02020202UL, + 0xc1c1c1c1UL, 0xafafafafUL, 0xbdbdbdbdUL, 0x03030303UL, + 0x01010101UL, 0x13131313UL, 0x8a8a8a8aUL, 0x6b6b6b6bUL, + 0x3a3a3a3aUL, 0x91919191UL, 0x11111111UL, 0x41414141UL, + 0x4f4f4f4fUL, 0x67676767UL, 0xdcdcdcdcUL, 0xeaeaeaeaUL, + 0x97979797UL, 0xf2f2f2f2UL, 0xcfcfcfcfUL, 0xcecececeUL, + 0xf0f0f0f0UL, 0xb4b4b4b4UL, 0xe6e6e6e6UL, 0x73737373UL, + 0x96969696UL, 0xacacacacUL, 0x74747474UL, 0x22222222UL, + 0xe7e7e7e7UL, 0xadadadadUL, 0x35353535UL, 0x85858585UL, + 0xe2e2e2e2UL, 0xf9f9f9f9UL, 0x37373737UL, 0xe8e8e8e8UL, + 0x1c1c1c1cUL, 0x75757575UL, 0xdfdfdfdfUL, 0x6e6e6e6eUL, + 0x47474747UL, 0xf1f1f1f1UL, 0x1a1a1a1aUL, 0x71717171UL, + 0x1d1d1d1dUL, 0x29292929UL, 0xc5c5c5c5UL, 0x89898989UL, + 0x6f6f6f6fUL, 0xb7b7b7b7UL, 0x62626262UL, 0x0e0e0e0eUL, + 0xaaaaaaaaUL, 0x18181818UL, 0xbebebebeUL, 0x1b1b1b1bUL, + 0xfcfcfcfcUL, 0x56565656UL, 0x3e3e3e3eUL, 0x4b4b4b4bUL, + 0xc6c6c6c6UL, 0xd2d2d2d2UL, 0x79797979UL, 0x20202020UL, + 0x9a9a9a9aUL, 0xdbdbdbdbUL, 0xc0c0c0c0UL, 0xfefefefeUL, + 0x78787878UL, 0xcdcdcdcdUL, 0x5a5a5a5aUL, 0xf4f4f4f4UL, + 0x1f1f1f1fUL, 0xddddddddUL, 0xa8a8a8a8UL, 0x33333333UL, + 0x88888888UL, 0x07070707UL, 0xc7c7c7c7UL, 0x31313131UL, + 0xb1b1b1b1UL, 0x12121212UL, 0x10101010UL, 0x59595959UL, + 0x27272727UL, 0x80808080UL, 0xececececUL, 0x5f5f5f5fUL, + 0x60606060UL, 0x51515151UL, 0x7f7f7f7fUL, 0xa9a9a9a9UL, + 0x19191919UL, 0xb5b5b5b5UL, 0x4a4a4a4aUL, 0x0d0d0d0dUL, + 0x2d2d2d2dUL, 0xe5e5e5e5UL, 0x7a7a7a7aUL, 0x9f9f9f9fUL, + 0x93939393UL, 0xc9c9c9c9UL, 0x9c9c9c9cUL, 0xefefefefUL, + 0xa0a0a0a0UL, 0xe0e0e0e0UL, 0x3b3b3b3bUL, 0x4d4d4d4dUL, + 0xaeaeaeaeUL, 0x2a2a2a2aUL, 0xf5f5f5f5UL, 0xb0b0b0b0UL, + 0xc8c8c8c8UL, 0xebebebebUL, 0xbbbbbbbbUL, 0x3c3c3c3cUL, + 0x83838383UL, 0x53535353UL, 0x99999999UL, 0x61616161UL, + 0x17171717UL, 0x2b2b2b2bUL, 0x04040404UL, 0x7e7e7e7eUL, + 0xbabababaUL, 0x77777777UL, 0xd6d6d6d6UL, 0x26262626UL, + 0xe1e1e1e1UL, 0x69696969UL, 0x14141414UL, 0x63636363UL, + 0x55555555UL, 0x21212121UL, 0x0c0c0c0cUL, 0x7d7d7d7dUL, +}; + +#endif /* ENCRYPT_ONLY */ + +#ifdef LTC_SMALL_CODE + +#define Te0(x) TE0[x] +#define Te1(x) RORc(TE0[x], 8) +#define Te2(x) RORc(TE0[x], 16) +#define Te3(x) RORc(TE0[x], 24) + +#define Td0(x) TD0[x] +#define Td1(x) RORc(TD0[x], 8) +#define Td2(x) RORc(TD0[x], 16) +#define Td3(x) RORc(TD0[x], 24) + +#define Te4_0 0x000000FF & Te4 +#define Te4_1 0x0000FF00 & Te4 +#define Te4_2 0x00FF0000 & Te4 +#define Te4_3 0xFF000000 & Te4 + +#else + +#define Te0(x) TE0[x] +#define Te1(x) TE1[x] +#define Te2(x) TE2[x] +#define Te3(x) TE3[x] + +#define Td0(x) TD0[x] +#define Td1(x) TD1[x] +#define Td2(x) TD2[x] +#define Td3(x) TD3[x] + +static const ulong32 TE1[256] = { + 0xa5c66363UL, 0x84f87c7cUL, 0x99ee7777UL, 0x8df67b7bUL, + 0x0dfff2f2UL, 0xbdd66b6bUL, 0xb1de6f6fUL, 0x5491c5c5UL, + 0x50603030UL, 0x03020101UL, 0xa9ce6767UL, 0x7d562b2bUL, + 0x19e7fefeUL, 0x62b5d7d7UL, 0xe64dababUL, 0x9aec7676UL, + 0x458fcacaUL, 0x9d1f8282UL, 0x4089c9c9UL, 0x87fa7d7dUL, + 0x15effafaUL, 0xebb25959UL, 0xc98e4747UL, 0x0bfbf0f0UL, + 0xec41adadUL, 0x67b3d4d4UL, 0xfd5fa2a2UL, 0xea45afafUL, + 0xbf239c9cUL, 0xf753a4a4UL, 0x96e47272UL, 0x5b9bc0c0UL, + 0xc275b7b7UL, 0x1ce1fdfdUL, 0xae3d9393UL, 0x6a4c2626UL, + 0x5a6c3636UL, 0x417e3f3fUL, 0x02f5f7f7UL, 0x4f83ccccUL, + 0x5c683434UL, 0xf451a5a5UL, 0x34d1e5e5UL, 0x08f9f1f1UL, + 0x93e27171UL, 0x73abd8d8UL, 0x53623131UL, 0x3f2a1515UL, + 0x0c080404UL, 0x5295c7c7UL, 0x65462323UL, 0x5e9dc3c3UL, + 0x28301818UL, 0xa1379696UL, 0x0f0a0505UL, 0xb52f9a9aUL, + 0x090e0707UL, 0x36241212UL, 0x9b1b8080UL, 0x3ddfe2e2UL, + 0x26cdebebUL, 0x694e2727UL, 0xcd7fb2b2UL, 0x9fea7575UL, + 0x1b120909UL, 0x9e1d8383UL, 0x74582c2cUL, 0x2e341a1aUL, + 0x2d361b1bUL, 0xb2dc6e6eUL, 0xeeb45a5aUL, 0xfb5ba0a0UL, + 0xf6a45252UL, 0x4d763b3bUL, 0x61b7d6d6UL, 0xce7db3b3UL, + 0x7b522929UL, 0x3edde3e3UL, 0x715e2f2fUL, 0x97138484UL, + 0xf5a65353UL, 0x68b9d1d1UL, 0x00000000UL, 0x2cc1ededUL, + 0x60402020UL, 0x1fe3fcfcUL, 0xc879b1b1UL, 0xedb65b5bUL, + 0xbed46a6aUL, 0x468dcbcbUL, 0xd967bebeUL, 0x4b723939UL, + 0xde944a4aUL, 0xd4984c4cUL, 0xe8b05858UL, 0x4a85cfcfUL, + 0x6bbbd0d0UL, 0x2ac5efefUL, 0xe54faaaaUL, 0x16edfbfbUL, + 0xc5864343UL, 0xd79a4d4dUL, 0x55663333UL, 0x94118585UL, + 0xcf8a4545UL, 0x10e9f9f9UL, 0x06040202UL, 0x81fe7f7fUL, + 0xf0a05050UL, 0x44783c3cUL, 0xba259f9fUL, 0xe34ba8a8UL, + 0xf3a25151UL, 0xfe5da3a3UL, 0xc0804040UL, 0x8a058f8fUL, + 0xad3f9292UL, 0xbc219d9dUL, 0x48703838UL, 0x04f1f5f5UL, + 0xdf63bcbcUL, 0xc177b6b6UL, 0x75afdadaUL, 0x63422121UL, + 0x30201010UL, 0x1ae5ffffUL, 0x0efdf3f3UL, 0x6dbfd2d2UL, + 0x4c81cdcdUL, 0x14180c0cUL, 0x35261313UL, 0x2fc3ececUL, + 0xe1be5f5fUL, 0xa2359797UL, 0xcc884444UL, 0x392e1717UL, + 0x5793c4c4UL, 0xf255a7a7UL, 0x82fc7e7eUL, 0x477a3d3dUL, + 0xacc86464UL, 0xe7ba5d5dUL, 0x2b321919UL, 0x95e67373UL, + 0xa0c06060UL, 0x98198181UL, 0xd19e4f4fUL, 0x7fa3dcdcUL, + 0x66442222UL, 0x7e542a2aUL, 0xab3b9090UL, 0x830b8888UL, + 0xca8c4646UL, 0x29c7eeeeUL, 0xd36bb8b8UL, 0x3c281414UL, + 0x79a7dedeUL, 0xe2bc5e5eUL, 0x1d160b0bUL, 0x76addbdbUL, + 0x3bdbe0e0UL, 0x56643232UL, 0x4e743a3aUL, 0x1e140a0aUL, + 0xdb924949UL, 0x0a0c0606UL, 0x6c482424UL, 0xe4b85c5cUL, + 0x5d9fc2c2UL, 0x6ebdd3d3UL, 0xef43acacUL, 0xa6c46262UL, + 0xa8399191UL, 0xa4319595UL, 0x37d3e4e4UL, 0x8bf27979UL, + 0x32d5e7e7UL, 0x438bc8c8UL, 0x596e3737UL, 0xb7da6d6dUL, + 0x8c018d8dUL, 0x64b1d5d5UL, 0xd29c4e4eUL, 0xe049a9a9UL, + 0xb4d86c6cUL, 0xfaac5656UL, 0x07f3f4f4UL, 0x25cfeaeaUL, + 0xafca6565UL, 0x8ef47a7aUL, 0xe947aeaeUL, 0x18100808UL, + 0xd56fbabaUL, 0x88f07878UL, 0x6f4a2525UL, 0x725c2e2eUL, + 0x24381c1cUL, 0xf157a6a6UL, 0xc773b4b4UL, 0x5197c6c6UL, + 0x23cbe8e8UL, 0x7ca1ddddUL, 0x9ce87474UL, 0x213e1f1fUL, + 0xdd964b4bUL, 0xdc61bdbdUL, 0x860d8b8bUL, 0x850f8a8aUL, + 0x90e07070UL, 0x427c3e3eUL, 0xc471b5b5UL, 0xaacc6666UL, + 0xd8904848UL, 0x05060303UL, 0x01f7f6f6UL, 0x121c0e0eUL, + 0xa3c26161UL, 0x5f6a3535UL, 0xf9ae5757UL, 0xd069b9b9UL, + 0x91178686UL, 0x5899c1c1UL, 0x273a1d1dUL, 0xb9279e9eUL, + 0x38d9e1e1UL, 0x13ebf8f8UL, 0xb32b9898UL, 0x33221111UL, + 0xbbd26969UL, 0x70a9d9d9UL, 0x89078e8eUL, 0xa7339494UL, + 0xb62d9b9bUL, 0x223c1e1eUL, 0x92158787UL, 0x20c9e9e9UL, + 0x4987ceceUL, 0xffaa5555UL, 0x78502828UL, 0x7aa5dfdfUL, + 0x8f038c8cUL, 0xf859a1a1UL, 0x80098989UL, 0x171a0d0dUL, + 0xda65bfbfUL, 0x31d7e6e6UL, 0xc6844242UL, 0xb8d06868UL, + 0xc3824141UL, 0xb0299999UL, 0x775a2d2dUL, 0x111e0f0fUL, + 0xcb7bb0b0UL, 0xfca85454UL, 0xd66dbbbbUL, 0x3a2c1616UL, +}; +static const ulong32 TE2[256] = { + 0x63a5c663UL, 0x7c84f87cUL, 0x7799ee77UL, 0x7b8df67bUL, + 0xf20dfff2UL, 0x6bbdd66bUL, 0x6fb1de6fUL, 0xc55491c5UL, + 0x30506030UL, 0x01030201UL, 0x67a9ce67UL, 0x2b7d562bUL, + 0xfe19e7feUL, 0xd762b5d7UL, 0xabe64dabUL, 0x769aec76UL, + 0xca458fcaUL, 0x829d1f82UL, 0xc94089c9UL, 0x7d87fa7dUL, + 0xfa15effaUL, 0x59ebb259UL, 0x47c98e47UL, 0xf00bfbf0UL, + 0xadec41adUL, 0xd467b3d4UL, 0xa2fd5fa2UL, 0xafea45afUL, + 0x9cbf239cUL, 0xa4f753a4UL, 0x7296e472UL, 0xc05b9bc0UL, + 0xb7c275b7UL, 0xfd1ce1fdUL, 0x93ae3d93UL, 0x266a4c26UL, + 0x365a6c36UL, 0x3f417e3fUL, 0xf702f5f7UL, 0xcc4f83ccUL, + 0x345c6834UL, 0xa5f451a5UL, 0xe534d1e5UL, 0xf108f9f1UL, + 0x7193e271UL, 0xd873abd8UL, 0x31536231UL, 0x153f2a15UL, + 0x040c0804UL, 0xc75295c7UL, 0x23654623UL, 0xc35e9dc3UL, + 0x18283018UL, 0x96a13796UL, 0x050f0a05UL, 0x9ab52f9aUL, + 0x07090e07UL, 0x12362412UL, 0x809b1b80UL, 0xe23ddfe2UL, + 0xeb26cdebUL, 0x27694e27UL, 0xb2cd7fb2UL, 0x759fea75UL, + 0x091b1209UL, 0x839e1d83UL, 0x2c74582cUL, 0x1a2e341aUL, + 0x1b2d361bUL, 0x6eb2dc6eUL, 0x5aeeb45aUL, 0xa0fb5ba0UL, + 0x52f6a452UL, 0x3b4d763bUL, 0xd661b7d6UL, 0xb3ce7db3UL, + 0x297b5229UL, 0xe33edde3UL, 0x2f715e2fUL, 0x84971384UL, + 0x53f5a653UL, 0xd168b9d1UL, 0x00000000UL, 0xed2cc1edUL, + 0x20604020UL, 0xfc1fe3fcUL, 0xb1c879b1UL, 0x5bedb65bUL, + 0x6abed46aUL, 0xcb468dcbUL, 0xbed967beUL, 0x394b7239UL, + 0x4ade944aUL, 0x4cd4984cUL, 0x58e8b058UL, 0xcf4a85cfUL, + 0xd06bbbd0UL, 0xef2ac5efUL, 0xaae54faaUL, 0xfb16edfbUL, + 0x43c58643UL, 0x4dd79a4dUL, 0x33556633UL, 0x85941185UL, + 0x45cf8a45UL, 0xf910e9f9UL, 0x02060402UL, 0x7f81fe7fUL, + 0x50f0a050UL, 0x3c44783cUL, 0x9fba259fUL, 0xa8e34ba8UL, + 0x51f3a251UL, 0xa3fe5da3UL, 0x40c08040UL, 0x8f8a058fUL, + 0x92ad3f92UL, 0x9dbc219dUL, 0x38487038UL, 0xf504f1f5UL, + 0xbcdf63bcUL, 0xb6c177b6UL, 0xda75afdaUL, 0x21634221UL, + 0x10302010UL, 0xff1ae5ffUL, 0xf30efdf3UL, 0xd26dbfd2UL, + 0xcd4c81cdUL, 0x0c14180cUL, 0x13352613UL, 0xec2fc3ecUL, + 0x5fe1be5fUL, 0x97a23597UL, 0x44cc8844UL, 0x17392e17UL, + 0xc45793c4UL, 0xa7f255a7UL, 0x7e82fc7eUL, 0x3d477a3dUL, + 0x64acc864UL, 0x5de7ba5dUL, 0x192b3219UL, 0x7395e673UL, + 0x60a0c060UL, 0x81981981UL, 0x4fd19e4fUL, 0xdc7fa3dcUL, + 0x22664422UL, 0x2a7e542aUL, 0x90ab3b90UL, 0x88830b88UL, + 0x46ca8c46UL, 0xee29c7eeUL, 0xb8d36bb8UL, 0x143c2814UL, + 0xde79a7deUL, 0x5ee2bc5eUL, 0x0b1d160bUL, 0xdb76addbUL, + 0xe03bdbe0UL, 0x32566432UL, 0x3a4e743aUL, 0x0a1e140aUL, + 0x49db9249UL, 0x060a0c06UL, 0x246c4824UL, 0x5ce4b85cUL, + 0xc25d9fc2UL, 0xd36ebdd3UL, 0xacef43acUL, 0x62a6c462UL, + 0x91a83991UL, 0x95a43195UL, 0xe437d3e4UL, 0x798bf279UL, + 0xe732d5e7UL, 0xc8438bc8UL, 0x37596e37UL, 0x6db7da6dUL, + 0x8d8c018dUL, 0xd564b1d5UL, 0x4ed29c4eUL, 0xa9e049a9UL, + 0x6cb4d86cUL, 0x56faac56UL, 0xf407f3f4UL, 0xea25cfeaUL, + 0x65afca65UL, 0x7a8ef47aUL, 0xaee947aeUL, 0x08181008UL, + 0xbad56fbaUL, 0x7888f078UL, 0x256f4a25UL, 0x2e725c2eUL, + 0x1c24381cUL, 0xa6f157a6UL, 0xb4c773b4UL, 0xc65197c6UL, + 0xe823cbe8UL, 0xdd7ca1ddUL, 0x749ce874UL, 0x1f213e1fUL, + 0x4bdd964bUL, 0xbddc61bdUL, 0x8b860d8bUL, 0x8a850f8aUL, + 0x7090e070UL, 0x3e427c3eUL, 0xb5c471b5UL, 0x66aacc66UL, + 0x48d89048UL, 0x03050603UL, 0xf601f7f6UL, 0x0e121c0eUL, + 0x61a3c261UL, 0x355f6a35UL, 0x57f9ae57UL, 0xb9d069b9UL, + 0x86911786UL, 0xc15899c1UL, 0x1d273a1dUL, 0x9eb9279eUL, + 0xe138d9e1UL, 0xf813ebf8UL, 0x98b32b98UL, 0x11332211UL, + 0x69bbd269UL, 0xd970a9d9UL, 0x8e89078eUL, 0x94a73394UL, + 0x9bb62d9bUL, 0x1e223c1eUL, 0x87921587UL, 0xe920c9e9UL, + 0xce4987ceUL, 0x55ffaa55UL, 0x28785028UL, 0xdf7aa5dfUL, + 0x8c8f038cUL, 0xa1f859a1UL, 0x89800989UL, 0x0d171a0dUL, + 0xbfda65bfUL, 0xe631d7e6UL, 0x42c68442UL, 0x68b8d068UL, + 0x41c38241UL, 0x99b02999UL, 0x2d775a2dUL, 0x0f111e0fUL, + 0xb0cb7bb0UL, 0x54fca854UL, 0xbbd66dbbUL, 0x163a2c16UL, +}; +static const ulong32 TE3[256] = { + + 0x6363a5c6UL, 0x7c7c84f8UL, 0x777799eeUL, 0x7b7b8df6UL, + 0xf2f20dffUL, 0x6b6bbdd6UL, 0x6f6fb1deUL, 0xc5c55491UL, + 0x30305060UL, 0x01010302UL, 0x6767a9ceUL, 0x2b2b7d56UL, + 0xfefe19e7UL, 0xd7d762b5UL, 0xababe64dUL, 0x76769aecUL, + 0xcaca458fUL, 0x82829d1fUL, 0xc9c94089UL, 0x7d7d87faUL, + 0xfafa15efUL, 0x5959ebb2UL, 0x4747c98eUL, 0xf0f00bfbUL, + 0xadadec41UL, 0xd4d467b3UL, 0xa2a2fd5fUL, 0xafafea45UL, + 0x9c9cbf23UL, 0xa4a4f753UL, 0x727296e4UL, 0xc0c05b9bUL, + 0xb7b7c275UL, 0xfdfd1ce1UL, 0x9393ae3dUL, 0x26266a4cUL, + 0x36365a6cUL, 0x3f3f417eUL, 0xf7f702f5UL, 0xcccc4f83UL, + 0x34345c68UL, 0xa5a5f451UL, 0xe5e534d1UL, 0xf1f108f9UL, + 0x717193e2UL, 0xd8d873abUL, 0x31315362UL, 0x15153f2aUL, + 0x04040c08UL, 0xc7c75295UL, 0x23236546UL, 0xc3c35e9dUL, + 0x18182830UL, 0x9696a137UL, 0x05050f0aUL, 0x9a9ab52fUL, + 0x0707090eUL, 0x12123624UL, 0x80809b1bUL, 0xe2e23ddfUL, + 0xebeb26cdUL, 0x2727694eUL, 0xb2b2cd7fUL, 0x75759feaUL, + 0x09091b12UL, 0x83839e1dUL, 0x2c2c7458UL, 0x1a1a2e34UL, + 0x1b1b2d36UL, 0x6e6eb2dcUL, 0x5a5aeeb4UL, 0xa0a0fb5bUL, + 0x5252f6a4UL, 0x3b3b4d76UL, 0xd6d661b7UL, 0xb3b3ce7dUL, + 0x29297b52UL, 0xe3e33eddUL, 0x2f2f715eUL, 0x84849713UL, + 0x5353f5a6UL, 0xd1d168b9UL, 0x00000000UL, 0xeded2cc1UL, + 0x20206040UL, 0xfcfc1fe3UL, 0xb1b1c879UL, 0x5b5bedb6UL, + 0x6a6abed4UL, 0xcbcb468dUL, 0xbebed967UL, 0x39394b72UL, + 0x4a4ade94UL, 0x4c4cd498UL, 0x5858e8b0UL, 0xcfcf4a85UL, + 0xd0d06bbbUL, 0xefef2ac5UL, 0xaaaae54fUL, 0xfbfb16edUL, + 0x4343c586UL, 0x4d4dd79aUL, 0x33335566UL, 0x85859411UL, + 0x4545cf8aUL, 0xf9f910e9UL, 0x02020604UL, 0x7f7f81feUL, + 0x5050f0a0UL, 0x3c3c4478UL, 0x9f9fba25UL, 0xa8a8e34bUL, + 0x5151f3a2UL, 0xa3a3fe5dUL, 0x4040c080UL, 0x8f8f8a05UL, + 0x9292ad3fUL, 0x9d9dbc21UL, 0x38384870UL, 0xf5f504f1UL, + 0xbcbcdf63UL, 0xb6b6c177UL, 0xdada75afUL, 0x21216342UL, + 0x10103020UL, 0xffff1ae5UL, 0xf3f30efdUL, 0xd2d26dbfUL, + 0xcdcd4c81UL, 0x0c0c1418UL, 0x13133526UL, 0xecec2fc3UL, + 0x5f5fe1beUL, 0x9797a235UL, 0x4444cc88UL, 0x1717392eUL, + 0xc4c45793UL, 0xa7a7f255UL, 0x7e7e82fcUL, 0x3d3d477aUL, + 0x6464acc8UL, 0x5d5de7baUL, 0x19192b32UL, 0x737395e6UL, + 0x6060a0c0UL, 0x81819819UL, 0x4f4fd19eUL, 0xdcdc7fa3UL, + 0x22226644UL, 0x2a2a7e54UL, 0x9090ab3bUL, 0x8888830bUL, + 0x4646ca8cUL, 0xeeee29c7UL, 0xb8b8d36bUL, 0x14143c28UL, + 0xdede79a7UL, 0x5e5ee2bcUL, 0x0b0b1d16UL, 0xdbdb76adUL, + 0xe0e03bdbUL, 0x32325664UL, 0x3a3a4e74UL, 0x0a0a1e14UL, + 0x4949db92UL, 0x06060a0cUL, 0x24246c48UL, 0x5c5ce4b8UL, + 0xc2c25d9fUL, 0xd3d36ebdUL, 0xacacef43UL, 0x6262a6c4UL, + 0x9191a839UL, 0x9595a431UL, 0xe4e437d3UL, 0x79798bf2UL, + 0xe7e732d5UL, 0xc8c8438bUL, 0x3737596eUL, 0x6d6db7daUL, + 0x8d8d8c01UL, 0xd5d564b1UL, 0x4e4ed29cUL, 0xa9a9e049UL, + 0x6c6cb4d8UL, 0x5656faacUL, 0xf4f407f3UL, 0xeaea25cfUL, + 0x6565afcaUL, 0x7a7a8ef4UL, 0xaeaee947UL, 0x08081810UL, + 0xbabad56fUL, 0x787888f0UL, 0x25256f4aUL, 0x2e2e725cUL, + 0x1c1c2438UL, 0xa6a6f157UL, 0xb4b4c773UL, 0xc6c65197UL, + 0xe8e823cbUL, 0xdddd7ca1UL, 0x74749ce8UL, 0x1f1f213eUL, + 0x4b4bdd96UL, 0xbdbddc61UL, 0x8b8b860dUL, 0x8a8a850fUL, + 0x707090e0UL, 0x3e3e427cUL, 0xb5b5c471UL, 0x6666aaccUL, + 0x4848d890UL, 0x03030506UL, 0xf6f601f7UL, 0x0e0e121cUL, + 0x6161a3c2UL, 0x35355f6aUL, 0x5757f9aeUL, 0xb9b9d069UL, + 0x86869117UL, 0xc1c15899UL, 0x1d1d273aUL, 0x9e9eb927UL, + 0xe1e138d9UL, 0xf8f813ebUL, 0x9898b32bUL, 0x11113322UL, + 0x6969bbd2UL, 0xd9d970a9UL, 0x8e8e8907UL, 0x9494a733UL, + 0x9b9bb62dUL, 0x1e1e223cUL, 0x87879215UL, 0xe9e920c9UL, + 0xcece4987UL, 0x5555ffaaUL, 0x28287850UL, 0xdfdf7aa5UL, + 0x8c8c8f03UL, 0xa1a1f859UL, 0x89898009UL, 0x0d0d171aUL, + 0xbfbfda65UL, 0xe6e631d7UL, 0x4242c684UL, 0x6868b8d0UL, + 0x4141c382UL, 0x9999b029UL, 0x2d2d775aUL, 0x0f0f111eUL, + 0xb0b0cb7bUL, 0x5454fca8UL, 0xbbbbd66dUL, 0x16163a2cUL, +}; + +#ifndef PELI_TAB +static const ulong32 Te4_0[] = { +0x00000063UL, 0x0000007cUL, 0x00000077UL, 0x0000007bUL, 0x000000f2UL, 0x0000006bUL, 0x0000006fUL, 0x000000c5UL, +0x00000030UL, 0x00000001UL, 0x00000067UL, 0x0000002bUL, 0x000000feUL, 0x000000d7UL, 0x000000abUL, 0x00000076UL, +0x000000caUL, 0x00000082UL, 0x000000c9UL, 0x0000007dUL, 0x000000faUL, 0x00000059UL, 0x00000047UL, 0x000000f0UL, +0x000000adUL, 0x000000d4UL, 0x000000a2UL, 0x000000afUL, 0x0000009cUL, 0x000000a4UL, 0x00000072UL, 0x000000c0UL, +0x000000b7UL, 0x000000fdUL, 0x00000093UL, 0x00000026UL, 0x00000036UL, 0x0000003fUL, 0x000000f7UL, 0x000000ccUL, +0x00000034UL, 0x000000a5UL, 0x000000e5UL, 0x000000f1UL, 0x00000071UL, 0x000000d8UL, 0x00000031UL, 0x00000015UL, +0x00000004UL, 0x000000c7UL, 0x00000023UL, 0x000000c3UL, 0x00000018UL, 0x00000096UL, 0x00000005UL, 0x0000009aUL, +0x00000007UL, 0x00000012UL, 0x00000080UL, 0x000000e2UL, 0x000000ebUL, 0x00000027UL, 0x000000b2UL, 0x00000075UL, +0x00000009UL, 0x00000083UL, 0x0000002cUL, 0x0000001aUL, 0x0000001bUL, 0x0000006eUL, 0x0000005aUL, 0x000000a0UL, +0x00000052UL, 0x0000003bUL, 0x000000d6UL, 0x000000b3UL, 0x00000029UL, 0x000000e3UL, 0x0000002fUL, 0x00000084UL, +0x00000053UL, 0x000000d1UL, 0x00000000UL, 0x000000edUL, 0x00000020UL, 0x000000fcUL, 0x000000b1UL, 0x0000005bUL, +0x0000006aUL, 0x000000cbUL, 0x000000beUL, 0x00000039UL, 0x0000004aUL, 0x0000004cUL, 0x00000058UL, 0x000000cfUL, +0x000000d0UL, 0x000000efUL, 0x000000aaUL, 0x000000fbUL, 0x00000043UL, 0x0000004dUL, 0x00000033UL, 0x00000085UL, +0x00000045UL, 0x000000f9UL, 0x00000002UL, 0x0000007fUL, 0x00000050UL, 0x0000003cUL, 0x0000009fUL, 0x000000a8UL, +0x00000051UL, 0x000000a3UL, 0x00000040UL, 0x0000008fUL, 0x00000092UL, 0x0000009dUL, 0x00000038UL, 0x000000f5UL, +0x000000bcUL, 0x000000b6UL, 0x000000daUL, 0x00000021UL, 0x00000010UL, 0x000000ffUL, 0x000000f3UL, 0x000000d2UL, +0x000000cdUL, 0x0000000cUL, 0x00000013UL, 0x000000ecUL, 0x0000005fUL, 0x00000097UL, 0x00000044UL, 0x00000017UL, +0x000000c4UL, 0x000000a7UL, 0x0000007eUL, 0x0000003dUL, 0x00000064UL, 0x0000005dUL, 0x00000019UL, 0x00000073UL, +0x00000060UL, 0x00000081UL, 0x0000004fUL, 0x000000dcUL, 0x00000022UL, 0x0000002aUL, 0x00000090UL, 0x00000088UL, +0x00000046UL, 0x000000eeUL, 0x000000b8UL, 0x00000014UL, 0x000000deUL, 0x0000005eUL, 0x0000000bUL, 0x000000dbUL, +0x000000e0UL, 0x00000032UL, 0x0000003aUL, 0x0000000aUL, 0x00000049UL, 0x00000006UL, 0x00000024UL, 0x0000005cUL, +0x000000c2UL, 0x000000d3UL, 0x000000acUL, 0x00000062UL, 0x00000091UL, 0x00000095UL, 0x000000e4UL, 0x00000079UL, +0x000000e7UL, 0x000000c8UL, 0x00000037UL, 0x0000006dUL, 0x0000008dUL, 0x000000d5UL, 0x0000004eUL, 0x000000a9UL, +0x0000006cUL, 0x00000056UL, 0x000000f4UL, 0x000000eaUL, 0x00000065UL, 0x0000007aUL, 0x000000aeUL, 0x00000008UL, +0x000000baUL, 0x00000078UL, 0x00000025UL, 0x0000002eUL, 0x0000001cUL, 0x000000a6UL, 0x000000b4UL, 0x000000c6UL, +0x000000e8UL, 0x000000ddUL, 0x00000074UL, 0x0000001fUL, 0x0000004bUL, 0x000000bdUL, 0x0000008bUL, 0x0000008aUL, +0x00000070UL, 0x0000003eUL, 0x000000b5UL, 0x00000066UL, 0x00000048UL, 0x00000003UL, 0x000000f6UL, 0x0000000eUL, +0x00000061UL, 0x00000035UL, 0x00000057UL, 0x000000b9UL, 0x00000086UL, 0x000000c1UL, 0x0000001dUL, 0x0000009eUL, +0x000000e1UL, 0x000000f8UL, 0x00000098UL, 0x00000011UL, 0x00000069UL, 0x000000d9UL, 0x0000008eUL, 0x00000094UL, +0x0000009bUL, 0x0000001eUL, 0x00000087UL, 0x000000e9UL, 0x000000ceUL, 0x00000055UL, 0x00000028UL, 0x000000dfUL, +0x0000008cUL, 0x000000a1UL, 0x00000089UL, 0x0000000dUL, 0x000000bfUL, 0x000000e6UL, 0x00000042UL, 0x00000068UL, +0x00000041UL, 0x00000099UL, 0x0000002dUL, 0x0000000fUL, 0x000000b0UL, 0x00000054UL, 0x000000bbUL, 0x00000016UL +}; + +static const ulong32 Te4_1[] = { +0x00006300UL, 0x00007c00UL, 0x00007700UL, 0x00007b00UL, 0x0000f200UL, 0x00006b00UL, 0x00006f00UL, 0x0000c500UL, +0x00003000UL, 0x00000100UL, 0x00006700UL, 0x00002b00UL, 0x0000fe00UL, 0x0000d700UL, 0x0000ab00UL, 0x00007600UL, +0x0000ca00UL, 0x00008200UL, 0x0000c900UL, 0x00007d00UL, 0x0000fa00UL, 0x00005900UL, 0x00004700UL, 0x0000f000UL, +0x0000ad00UL, 0x0000d400UL, 0x0000a200UL, 0x0000af00UL, 0x00009c00UL, 0x0000a400UL, 0x00007200UL, 0x0000c000UL, +0x0000b700UL, 0x0000fd00UL, 0x00009300UL, 0x00002600UL, 0x00003600UL, 0x00003f00UL, 0x0000f700UL, 0x0000cc00UL, +0x00003400UL, 0x0000a500UL, 0x0000e500UL, 0x0000f100UL, 0x00007100UL, 0x0000d800UL, 0x00003100UL, 0x00001500UL, +0x00000400UL, 0x0000c700UL, 0x00002300UL, 0x0000c300UL, 0x00001800UL, 0x00009600UL, 0x00000500UL, 0x00009a00UL, +0x00000700UL, 0x00001200UL, 0x00008000UL, 0x0000e200UL, 0x0000eb00UL, 0x00002700UL, 0x0000b200UL, 0x00007500UL, +0x00000900UL, 0x00008300UL, 0x00002c00UL, 0x00001a00UL, 0x00001b00UL, 0x00006e00UL, 0x00005a00UL, 0x0000a000UL, +0x00005200UL, 0x00003b00UL, 0x0000d600UL, 0x0000b300UL, 0x00002900UL, 0x0000e300UL, 0x00002f00UL, 0x00008400UL, +0x00005300UL, 0x0000d100UL, 0x00000000UL, 0x0000ed00UL, 0x00002000UL, 0x0000fc00UL, 0x0000b100UL, 0x00005b00UL, +0x00006a00UL, 0x0000cb00UL, 0x0000be00UL, 0x00003900UL, 0x00004a00UL, 0x00004c00UL, 0x00005800UL, 0x0000cf00UL, +0x0000d000UL, 0x0000ef00UL, 0x0000aa00UL, 0x0000fb00UL, 0x00004300UL, 0x00004d00UL, 0x00003300UL, 0x00008500UL, +0x00004500UL, 0x0000f900UL, 0x00000200UL, 0x00007f00UL, 0x00005000UL, 0x00003c00UL, 0x00009f00UL, 0x0000a800UL, +0x00005100UL, 0x0000a300UL, 0x00004000UL, 0x00008f00UL, 0x00009200UL, 0x00009d00UL, 0x00003800UL, 0x0000f500UL, +0x0000bc00UL, 0x0000b600UL, 0x0000da00UL, 0x00002100UL, 0x00001000UL, 0x0000ff00UL, 0x0000f300UL, 0x0000d200UL, +0x0000cd00UL, 0x00000c00UL, 0x00001300UL, 0x0000ec00UL, 0x00005f00UL, 0x00009700UL, 0x00004400UL, 0x00001700UL, +0x0000c400UL, 0x0000a700UL, 0x00007e00UL, 0x00003d00UL, 0x00006400UL, 0x00005d00UL, 0x00001900UL, 0x00007300UL, +0x00006000UL, 0x00008100UL, 0x00004f00UL, 0x0000dc00UL, 0x00002200UL, 0x00002a00UL, 0x00009000UL, 0x00008800UL, +0x00004600UL, 0x0000ee00UL, 0x0000b800UL, 0x00001400UL, 0x0000de00UL, 0x00005e00UL, 0x00000b00UL, 0x0000db00UL, +0x0000e000UL, 0x00003200UL, 0x00003a00UL, 0x00000a00UL, 0x00004900UL, 0x00000600UL, 0x00002400UL, 0x00005c00UL, +0x0000c200UL, 0x0000d300UL, 0x0000ac00UL, 0x00006200UL, 0x00009100UL, 0x00009500UL, 0x0000e400UL, 0x00007900UL, +0x0000e700UL, 0x0000c800UL, 0x00003700UL, 0x00006d00UL, 0x00008d00UL, 0x0000d500UL, 0x00004e00UL, 0x0000a900UL, +0x00006c00UL, 0x00005600UL, 0x0000f400UL, 0x0000ea00UL, 0x00006500UL, 0x00007a00UL, 0x0000ae00UL, 0x00000800UL, +0x0000ba00UL, 0x00007800UL, 0x00002500UL, 0x00002e00UL, 0x00001c00UL, 0x0000a600UL, 0x0000b400UL, 0x0000c600UL, +0x0000e800UL, 0x0000dd00UL, 0x00007400UL, 0x00001f00UL, 0x00004b00UL, 0x0000bd00UL, 0x00008b00UL, 0x00008a00UL, +0x00007000UL, 0x00003e00UL, 0x0000b500UL, 0x00006600UL, 0x00004800UL, 0x00000300UL, 0x0000f600UL, 0x00000e00UL, +0x00006100UL, 0x00003500UL, 0x00005700UL, 0x0000b900UL, 0x00008600UL, 0x0000c100UL, 0x00001d00UL, 0x00009e00UL, +0x0000e100UL, 0x0000f800UL, 0x00009800UL, 0x00001100UL, 0x00006900UL, 0x0000d900UL, 0x00008e00UL, 0x00009400UL, +0x00009b00UL, 0x00001e00UL, 0x00008700UL, 0x0000e900UL, 0x0000ce00UL, 0x00005500UL, 0x00002800UL, 0x0000df00UL, +0x00008c00UL, 0x0000a100UL, 0x00008900UL, 0x00000d00UL, 0x0000bf00UL, 0x0000e600UL, 0x00004200UL, 0x00006800UL, +0x00004100UL, 0x00009900UL, 0x00002d00UL, 0x00000f00UL, 0x0000b000UL, 0x00005400UL, 0x0000bb00UL, 0x00001600UL +}; + +static const ulong32 Te4_2[] = { +0x00630000UL, 0x007c0000UL, 0x00770000UL, 0x007b0000UL, 0x00f20000UL, 0x006b0000UL, 0x006f0000UL, 0x00c50000UL, +0x00300000UL, 0x00010000UL, 0x00670000UL, 0x002b0000UL, 0x00fe0000UL, 0x00d70000UL, 0x00ab0000UL, 0x00760000UL, +0x00ca0000UL, 0x00820000UL, 0x00c90000UL, 0x007d0000UL, 0x00fa0000UL, 0x00590000UL, 0x00470000UL, 0x00f00000UL, +0x00ad0000UL, 0x00d40000UL, 0x00a20000UL, 0x00af0000UL, 0x009c0000UL, 0x00a40000UL, 0x00720000UL, 0x00c00000UL, +0x00b70000UL, 0x00fd0000UL, 0x00930000UL, 0x00260000UL, 0x00360000UL, 0x003f0000UL, 0x00f70000UL, 0x00cc0000UL, +0x00340000UL, 0x00a50000UL, 0x00e50000UL, 0x00f10000UL, 0x00710000UL, 0x00d80000UL, 0x00310000UL, 0x00150000UL, +0x00040000UL, 0x00c70000UL, 0x00230000UL, 0x00c30000UL, 0x00180000UL, 0x00960000UL, 0x00050000UL, 0x009a0000UL, +0x00070000UL, 0x00120000UL, 0x00800000UL, 0x00e20000UL, 0x00eb0000UL, 0x00270000UL, 0x00b20000UL, 0x00750000UL, +0x00090000UL, 0x00830000UL, 0x002c0000UL, 0x001a0000UL, 0x001b0000UL, 0x006e0000UL, 0x005a0000UL, 0x00a00000UL, +0x00520000UL, 0x003b0000UL, 0x00d60000UL, 0x00b30000UL, 0x00290000UL, 0x00e30000UL, 0x002f0000UL, 0x00840000UL, +0x00530000UL, 0x00d10000UL, 0x00000000UL, 0x00ed0000UL, 0x00200000UL, 0x00fc0000UL, 0x00b10000UL, 0x005b0000UL, +0x006a0000UL, 0x00cb0000UL, 0x00be0000UL, 0x00390000UL, 0x004a0000UL, 0x004c0000UL, 0x00580000UL, 0x00cf0000UL, +0x00d00000UL, 0x00ef0000UL, 0x00aa0000UL, 0x00fb0000UL, 0x00430000UL, 0x004d0000UL, 0x00330000UL, 0x00850000UL, +0x00450000UL, 0x00f90000UL, 0x00020000UL, 0x007f0000UL, 0x00500000UL, 0x003c0000UL, 0x009f0000UL, 0x00a80000UL, +0x00510000UL, 0x00a30000UL, 0x00400000UL, 0x008f0000UL, 0x00920000UL, 0x009d0000UL, 0x00380000UL, 0x00f50000UL, +0x00bc0000UL, 0x00b60000UL, 0x00da0000UL, 0x00210000UL, 0x00100000UL, 0x00ff0000UL, 0x00f30000UL, 0x00d20000UL, +0x00cd0000UL, 0x000c0000UL, 0x00130000UL, 0x00ec0000UL, 0x005f0000UL, 0x00970000UL, 0x00440000UL, 0x00170000UL, +0x00c40000UL, 0x00a70000UL, 0x007e0000UL, 0x003d0000UL, 0x00640000UL, 0x005d0000UL, 0x00190000UL, 0x00730000UL, +0x00600000UL, 0x00810000UL, 0x004f0000UL, 0x00dc0000UL, 0x00220000UL, 0x002a0000UL, 0x00900000UL, 0x00880000UL, +0x00460000UL, 0x00ee0000UL, 0x00b80000UL, 0x00140000UL, 0x00de0000UL, 0x005e0000UL, 0x000b0000UL, 0x00db0000UL, +0x00e00000UL, 0x00320000UL, 0x003a0000UL, 0x000a0000UL, 0x00490000UL, 0x00060000UL, 0x00240000UL, 0x005c0000UL, +0x00c20000UL, 0x00d30000UL, 0x00ac0000UL, 0x00620000UL, 0x00910000UL, 0x00950000UL, 0x00e40000UL, 0x00790000UL, +0x00e70000UL, 0x00c80000UL, 0x00370000UL, 0x006d0000UL, 0x008d0000UL, 0x00d50000UL, 0x004e0000UL, 0x00a90000UL, +0x006c0000UL, 0x00560000UL, 0x00f40000UL, 0x00ea0000UL, 0x00650000UL, 0x007a0000UL, 0x00ae0000UL, 0x00080000UL, +0x00ba0000UL, 0x00780000UL, 0x00250000UL, 0x002e0000UL, 0x001c0000UL, 0x00a60000UL, 0x00b40000UL, 0x00c60000UL, +0x00e80000UL, 0x00dd0000UL, 0x00740000UL, 0x001f0000UL, 0x004b0000UL, 0x00bd0000UL, 0x008b0000UL, 0x008a0000UL, +0x00700000UL, 0x003e0000UL, 0x00b50000UL, 0x00660000UL, 0x00480000UL, 0x00030000UL, 0x00f60000UL, 0x000e0000UL, +0x00610000UL, 0x00350000UL, 0x00570000UL, 0x00b90000UL, 0x00860000UL, 0x00c10000UL, 0x001d0000UL, 0x009e0000UL, +0x00e10000UL, 0x00f80000UL, 0x00980000UL, 0x00110000UL, 0x00690000UL, 0x00d90000UL, 0x008e0000UL, 0x00940000UL, +0x009b0000UL, 0x001e0000UL, 0x00870000UL, 0x00e90000UL, 0x00ce0000UL, 0x00550000UL, 0x00280000UL, 0x00df0000UL, +0x008c0000UL, 0x00a10000UL, 0x00890000UL, 0x000d0000UL, 0x00bf0000UL, 0x00e60000UL, 0x00420000UL, 0x00680000UL, +0x00410000UL, 0x00990000UL, 0x002d0000UL, 0x000f0000UL, 0x00b00000UL, 0x00540000UL, 0x00bb0000UL, 0x00160000UL +}; + +static const ulong32 Te4_3[] = { +0x63000000UL, 0x7c000000UL, 0x77000000UL, 0x7b000000UL, 0xf2000000UL, 0x6b000000UL, 0x6f000000UL, 0xc5000000UL, +0x30000000UL, 0x01000000UL, 0x67000000UL, 0x2b000000UL, 0xfe000000UL, 0xd7000000UL, 0xab000000UL, 0x76000000UL, +0xca000000UL, 0x82000000UL, 0xc9000000UL, 0x7d000000UL, 0xfa000000UL, 0x59000000UL, 0x47000000UL, 0xf0000000UL, +0xad000000UL, 0xd4000000UL, 0xa2000000UL, 0xaf000000UL, 0x9c000000UL, 0xa4000000UL, 0x72000000UL, 0xc0000000UL, +0xb7000000UL, 0xfd000000UL, 0x93000000UL, 0x26000000UL, 0x36000000UL, 0x3f000000UL, 0xf7000000UL, 0xcc000000UL, +0x34000000UL, 0xa5000000UL, 0xe5000000UL, 0xf1000000UL, 0x71000000UL, 0xd8000000UL, 0x31000000UL, 0x15000000UL, +0x04000000UL, 0xc7000000UL, 0x23000000UL, 0xc3000000UL, 0x18000000UL, 0x96000000UL, 0x05000000UL, 0x9a000000UL, +0x07000000UL, 0x12000000UL, 0x80000000UL, 0xe2000000UL, 0xeb000000UL, 0x27000000UL, 0xb2000000UL, 0x75000000UL, +0x09000000UL, 0x83000000UL, 0x2c000000UL, 0x1a000000UL, 0x1b000000UL, 0x6e000000UL, 0x5a000000UL, 0xa0000000UL, +0x52000000UL, 0x3b000000UL, 0xd6000000UL, 0xb3000000UL, 0x29000000UL, 0xe3000000UL, 0x2f000000UL, 0x84000000UL, +0x53000000UL, 0xd1000000UL, 0x00000000UL, 0xed000000UL, 0x20000000UL, 0xfc000000UL, 0xb1000000UL, 0x5b000000UL, +0x6a000000UL, 0xcb000000UL, 0xbe000000UL, 0x39000000UL, 0x4a000000UL, 0x4c000000UL, 0x58000000UL, 0xcf000000UL, +0xd0000000UL, 0xef000000UL, 0xaa000000UL, 0xfb000000UL, 0x43000000UL, 0x4d000000UL, 0x33000000UL, 0x85000000UL, +0x45000000UL, 0xf9000000UL, 0x02000000UL, 0x7f000000UL, 0x50000000UL, 0x3c000000UL, 0x9f000000UL, 0xa8000000UL, +0x51000000UL, 0xa3000000UL, 0x40000000UL, 0x8f000000UL, 0x92000000UL, 0x9d000000UL, 0x38000000UL, 0xf5000000UL, +0xbc000000UL, 0xb6000000UL, 0xda000000UL, 0x21000000UL, 0x10000000UL, 0xff000000UL, 0xf3000000UL, 0xd2000000UL, +0xcd000000UL, 0x0c000000UL, 0x13000000UL, 0xec000000UL, 0x5f000000UL, 0x97000000UL, 0x44000000UL, 0x17000000UL, +0xc4000000UL, 0xa7000000UL, 0x7e000000UL, 0x3d000000UL, 0x64000000UL, 0x5d000000UL, 0x19000000UL, 0x73000000UL, +0x60000000UL, 0x81000000UL, 0x4f000000UL, 0xdc000000UL, 0x22000000UL, 0x2a000000UL, 0x90000000UL, 0x88000000UL, +0x46000000UL, 0xee000000UL, 0xb8000000UL, 0x14000000UL, 0xde000000UL, 0x5e000000UL, 0x0b000000UL, 0xdb000000UL, +0xe0000000UL, 0x32000000UL, 0x3a000000UL, 0x0a000000UL, 0x49000000UL, 0x06000000UL, 0x24000000UL, 0x5c000000UL, +0xc2000000UL, 0xd3000000UL, 0xac000000UL, 0x62000000UL, 0x91000000UL, 0x95000000UL, 0xe4000000UL, 0x79000000UL, +0xe7000000UL, 0xc8000000UL, 0x37000000UL, 0x6d000000UL, 0x8d000000UL, 0xd5000000UL, 0x4e000000UL, 0xa9000000UL, +0x6c000000UL, 0x56000000UL, 0xf4000000UL, 0xea000000UL, 0x65000000UL, 0x7a000000UL, 0xae000000UL, 0x08000000UL, +0xba000000UL, 0x78000000UL, 0x25000000UL, 0x2e000000UL, 0x1c000000UL, 0xa6000000UL, 0xb4000000UL, 0xc6000000UL, +0xe8000000UL, 0xdd000000UL, 0x74000000UL, 0x1f000000UL, 0x4b000000UL, 0xbd000000UL, 0x8b000000UL, 0x8a000000UL, +0x70000000UL, 0x3e000000UL, 0xb5000000UL, 0x66000000UL, 0x48000000UL, 0x03000000UL, 0xf6000000UL, 0x0e000000UL, +0x61000000UL, 0x35000000UL, 0x57000000UL, 0xb9000000UL, 0x86000000UL, 0xc1000000UL, 0x1d000000UL, 0x9e000000UL, +0xe1000000UL, 0xf8000000UL, 0x98000000UL, 0x11000000UL, 0x69000000UL, 0xd9000000UL, 0x8e000000UL, 0x94000000UL, +0x9b000000UL, 0x1e000000UL, 0x87000000UL, 0xe9000000UL, 0xce000000UL, 0x55000000UL, 0x28000000UL, 0xdf000000UL, +0x8c000000UL, 0xa1000000UL, 0x89000000UL, 0x0d000000UL, 0xbf000000UL, 0xe6000000UL, 0x42000000UL, 0x68000000UL, +0x41000000UL, 0x99000000UL, 0x2d000000UL, 0x0f000000UL, 0xb0000000UL, 0x54000000UL, 0xbb000000UL, 0x16000000UL +}; +#endif /* pelimac */ + +#ifndef ENCRYPT_ONLY + +static const ulong32 TD1[256] = { + 0x5051f4a7UL, 0x537e4165UL, 0xc31a17a4UL, 0x963a275eUL, + 0xcb3bab6bUL, 0xf11f9d45UL, 0xabacfa58UL, 0x934be303UL, + 0x552030faUL, 0xf6ad766dUL, 0x9188cc76UL, 0x25f5024cUL, + 0xfc4fe5d7UL, 0xd7c52acbUL, 0x80263544UL, 0x8fb562a3UL, + 0x49deb15aUL, 0x6725ba1bUL, 0x9845ea0eUL, 0xe15dfec0UL, + 0x02c32f75UL, 0x12814cf0UL, 0xa38d4697UL, 0xc66bd3f9UL, + 0xe7038f5fUL, 0x9515929cUL, 0xebbf6d7aUL, 0xda955259UL, + 0x2dd4be83UL, 0xd3587421UL, 0x2949e069UL, 0x448ec9c8UL, + 0x6a75c289UL, 0x78f48e79UL, 0x6b99583eUL, 0xdd27b971UL, + 0xb6bee14fUL, 0x17f088adUL, 0x66c920acUL, 0xb47dce3aUL, + 0x1863df4aUL, 0x82e51a31UL, 0x60975133UL, 0x4562537fUL, + 0xe0b16477UL, 0x84bb6baeUL, 0x1cfe81a0UL, 0x94f9082bUL, + 0x58704868UL, 0x198f45fdUL, 0x8794de6cUL, 0xb7527bf8UL, + 0x23ab73d3UL, 0xe2724b02UL, 0x57e31f8fUL, 0x2a6655abUL, + 0x07b2eb28UL, 0x032fb5c2UL, 0x9a86c57bUL, 0xa5d33708UL, + 0xf2302887UL, 0xb223bfa5UL, 0xba02036aUL, 0x5ced1682UL, + 0x2b8acf1cUL, 0x92a779b4UL, 0xf0f307f2UL, 0xa14e69e2UL, + 0xcd65daf4UL, 0xd50605beUL, 0x1fd13462UL, 0x8ac4a6feUL, + 0x9d342e53UL, 0xa0a2f355UL, 0x32058ae1UL, 0x75a4f6ebUL, + 0x390b83ecUL, 0xaa4060efUL, 0x065e719fUL, 0x51bd6e10UL, + 0xf93e218aUL, 0x3d96dd06UL, 0xaedd3e05UL, 0x464de6bdUL, + 0xb591548dUL, 0x0571c45dUL, 0x6f0406d4UL, 0xff605015UL, + 0x241998fbUL, 0x97d6bde9UL, 0xcc894043UL, 0x7767d99eUL, + 0xbdb0e842UL, 0x8807898bUL, 0x38e7195bUL, 0xdb79c8eeUL, + 0x47a17c0aUL, 0xe97c420fUL, 0xc9f8841eUL, 0x00000000UL, + 0x83098086UL, 0x48322bedUL, 0xac1e1170UL, 0x4e6c5a72UL, + 0xfbfd0effUL, 0x560f8538UL, 0x1e3daed5UL, 0x27362d39UL, + 0x640a0fd9UL, 0x21685ca6UL, 0xd19b5b54UL, 0x3a24362eUL, + 0xb10c0a67UL, 0x0f9357e7UL, 0xd2b4ee96UL, 0x9e1b9b91UL, + 0x4f80c0c5UL, 0xa261dc20UL, 0x695a774bUL, 0x161c121aUL, + 0x0ae293baUL, 0xe5c0a02aUL, 0x433c22e0UL, 0x1d121b17UL, + 0x0b0e090dUL, 0xadf28bc7UL, 0xb92db6a8UL, 0xc8141ea9UL, + 0x8557f119UL, 0x4caf7507UL, 0xbbee99ddUL, 0xfda37f60UL, + 0x9ff70126UL, 0xbc5c72f5UL, 0xc544663bUL, 0x345bfb7eUL, + 0x768b4329UL, 0xdccb23c6UL, 0x68b6edfcUL, 0x63b8e4f1UL, + 0xcad731dcUL, 0x10426385UL, 0x40139722UL, 0x2084c611UL, + 0x7d854a24UL, 0xf8d2bb3dUL, 0x11aef932UL, 0x6dc729a1UL, + 0x4b1d9e2fUL, 0xf3dcb230UL, 0xec0d8652UL, 0xd077c1e3UL, + 0x6c2bb316UL, 0x99a970b9UL, 0xfa119448UL, 0x2247e964UL, + 0xc4a8fc8cUL, 0x1aa0f03fUL, 0xd8567d2cUL, 0xef223390UL, + 0xc787494eUL, 0xc1d938d1UL, 0xfe8ccaa2UL, 0x3698d40bUL, + 0xcfa6f581UL, 0x28a57adeUL, 0x26dab78eUL, 0xa43fadbfUL, + 0xe42c3a9dUL, 0x0d507892UL, 0x9b6a5fccUL, 0x62547e46UL, + 0xc2f68d13UL, 0xe890d8b8UL, 0x5e2e39f7UL, 0xf582c3afUL, + 0xbe9f5d80UL, 0x7c69d093UL, 0xa96fd52dUL, 0xb3cf2512UL, + 0x3bc8ac99UL, 0xa710187dUL, 0x6ee89c63UL, 0x7bdb3bbbUL, + 0x09cd2678UL, 0xf46e5918UL, 0x01ec9ab7UL, 0xa8834f9aUL, + 0x65e6956eUL, 0x7eaaffe6UL, 0x0821bccfUL, 0xe6ef15e8UL, + 0xd9bae79bUL, 0xce4a6f36UL, 0xd4ea9f09UL, 0xd629b07cUL, + 0xaf31a4b2UL, 0x312a3f23UL, 0x30c6a594UL, 0xc035a266UL, + 0x37744ebcUL, 0xa6fc82caUL, 0xb0e090d0UL, 0x1533a7d8UL, + 0x4af10498UL, 0xf741ecdaUL, 0x0e7fcd50UL, 0x2f1791f6UL, + 0x8d764dd6UL, 0x4d43efb0UL, 0x54ccaa4dUL, 0xdfe49604UL, + 0xe39ed1b5UL, 0x1b4c6a88UL, 0xb8c12c1fUL, 0x7f466551UL, + 0x049d5eeaUL, 0x5d018c35UL, 0x73fa8774UL, 0x2efb0b41UL, + 0x5ab3671dUL, 0x5292dbd2UL, 0x33e91056UL, 0x136dd647UL, + 0x8c9ad761UL, 0x7a37a10cUL, 0x8e59f814UL, 0x89eb133cUL, + 0xeecea927UL, 0x35b761c9UL, 0xede11ce5UL, 0x3c7a47b1UL, + 0x599cd2dfUL, 0x3f55f273UL, 0x791814ceUL, 0xbf73c737UL, + 0xea53f7cdUL, 0x5b5ffdaaUL, 0x14df3d6fUL, 0x867844dbUL, + 0x81caaff3UL, 0x3eb968c4UL, 0x2c382434UL, 0x5fc2a340UL, + 0x72161dc3UL, 0x0cbce225UL, 0x8b283c49UL, 0x41ff0d95UL, + 0x7139a801UL, 0xde080cb3UL, 0x9cd8b4e4UL, 0x906456c1UL, + 0x617bcb84UL, 0x70d532b6UL, 0x74486c5cUL, 0x42d0b857UL, +}; +static const ulong32 TD2[256] = { + 0xa75051f4UL, 0x65537e41UL, 0xa4c31a17UL, 0x5e963a27UL, + 0x6bcb3babUL, 0x45f11f9dUL, 0x58abacfaUL, 0x03934be3UL, + 0xfa552030UL, 0x6df6ad76UL, 0x769188ccUL, 0x4c25f502UL, + 0xd7fc4fe5UL, 0xcbd7c52aUL, 0x44802635UL, 0xa38fb562UL, + 0x5a49deb1UL, 0x1b6725baUL, 0x0e9845eaUL, 0xc0e15dfeUL, + 0x7502c32fUL, 0xf012814cUL, 0x97a38d46UL, 0xf9c66bd3UL, + 0x5fe7038fUL, 0x9c951592UL, 0x7aebbf6dUL, 0x59da9552UL, + 0x832dd4beUL, 0x21d35874UL, 0x692949e0UL, 0xc8448ec9UL, + 0x896a75c2UL, 0x7978f48eUL, 0x3e6b9958UL, 0x71dd27b9UL, + 0x4fb6bee1UL, 0xad17f088UL, 0xac66c920UL, 0x3ab47dceUL, + 0x4a1863dfUL, 0x3182e51aUL, 0x33609751UL, 0x7f456253UL, + 0x77e0b164UL, 0xae84bb6bUL, 0xa01cfe81UL, 0x2b94f908UL, + 0x68587048UL, 0xfd198f45UL, 0x6c8794deUL, 0xf8b7527bUL, + 0xd323ab73UL, 0x02e2724bUL, 0x8f57e31fUL, 0xab2a6655UL, + 0x2807b2ebUL, 0xc2032fb5UL, 0x7b9a86c5UL, 0x08a5d337UL, + 0x87f23028UL, 0xa5b223bfUL, 0x6aba0203UL, 0x825ced16UL, + 0x1c2b8acfUL, 0xb492a779UL, 0xf2f0f307UL, 0xe2a14e69UL, + 0xf4cd65daUL, 0xbed50605UL, 0x621fd134UL, 0xfe8ac4a6UL, + 0x539d342eUL, 0x55a0a2f3UL, 0xe132058aUL, 0xeb75a4f6UL, + 0xec390b83UL, 0xefaa4060UL, 0x9f065e71UL, 0x1051bd6eUL, + 0x8af93e21UL, 0x063d96ddUL, 0x05aedd3eUL, 0xbd464de6UL, + 0x8db59154UL, 0x5d0571c4UL, 0xd46f0406UL, 0x15ff6050UL, + 0xfb241998UL, 0xe997d6bdUL, 0x43cc8940UL, 0x9e7767d9UL, + 0x42bdb0e8UL, 0x8b880789UL, 0x5b38e719UL, 0xeedb79c8UL, + 0x0a47a17cUL, 0x0fe97c42UL, 0x1ec9f884UL, 0x00000000UL, + 0x86830980UL, 0xed48322bUL, 0x70ac1e11UL, 0x724e6c5aUL, + 0xfffbfd0eUL, 0x38560f85UL, 0xd51e3daeUL, 0x3927362dUL, + 0xd9640a0fUL, 0xa621685cUL, 0x54d19b5bUL, 0x2e3a2436UL, + 0x67b10c0aUL, 0xe70f9357UL, 0x96d2b4eeUL, 0x919e1b9bUL, + 0xc54f80c0UL, 0x20a261dcUL, 0x4b695a77UL, 0x1a161c12UL, + 0xba0ae293UL, 0x2ae5c0a0UL, 0xe0433c22UL, 0x171d121bUL, + 0x0d0b0e09UL, 0xc7adf28bUL, 0xa8b92db6UL, 0xa9c8141eUL, + 0x198557f1UL, 0x074caf75UL, 0xddbbee99UL, 0x60fda37fUL, + 0x269ff701UL, 0xf5bc5c72UL, 0x3bc54466UL, 0x7e345bfbUL, + 0x29768b43UL, 0xc6dccb23UL, 0xfc68b6edUL, 0xf163b8e4UL, + 0xdccad731UL, 0x85104263UL, 0x22401397UL, 0x112084c6UL, + 0x247d854aUL, 0x3df8d2bbUL, 0x3211aef9UL, 0xa16dc729UL, + 0x2f4b1d9eUL, 0x30f3dcb2UL, 0x52ec0d86UL, 0xe3d077c1UL, + 0x166c2bb3UL, 0xb999a970UL, 0x48fa1194UL, 0x642247e9UL, + 0x8cc4a8fcUL, 0x3f1aa0f0UL, 0x2cd8567dUL, 0x90ef2233UL, + 0x4ec78749UL, 0xd1c1d938UL, 0xa2fe8ccaUL, 0x0b3698d4UL, + 0x81cfa6f5UL, 0xde28a57aUL, 0x8e26dab7UL, 0xbfa43fadUL, + 0x9de42c3aUL, 0x920d5078UL, 0xcc9b6a5fUL, 0x4662547eUL, + 0x13c2f68dUL, 0xb8e890d8UL, 0xf75e2e39UL, 0xaff582c3UL, + 0x80be9f5dUL, 0x937c69d0UL, 0x2da96fd5UL, 0x12b3cf25UL, + 0x993bc8acUL, 0x7da71018UL, 0x636ee89cUL, 0xbb7bdb3bUL, + 0x7809cd26UL, 0x18f46e59UL, 0xb701ec9aUL, 0x9aa8834fUL, + 0x6e65e695UL, 0xe67eaaffUL, 0xcf0821bcUL, 0xe8e6ef15UL, + 0x9bd9bae7UL, 0x36ce4a6fUL, 0x09d4ea9fUL, 0x7cd629b0UL, + 0xb2af31a4UL, 0x23312a3fUL, 0x9430c6a5UL, 0x66c035a2UL, + 0xbc37744eUL, 0xcaa6fc82UL, 0xd0b0e090UL, 0xd81533a7UL, + 0x984af104UL, 0xdaf741ecUL, 0x500e7fcdUL, 0xf62f1791UL, + 0xd68d764dUL, 0xb04d43efUL, 0x4d54ccaaUL, 0x04dfe496UL, + 0xb5e39ed1UL, 0x881b4c6aUL, 0x1fb8c12cUL, 0x517f4665UL, + 0xea049d5eUL, 0x355d018cUL, 0x7473fa87UL, 0x412efb0bUL, + 0x1d5ab367UL, 0xd25292dbUL, 0x5633e910UL, 0x47136dd6UL, + 0x618c9ad7UL, 0x0c7a37a1UL, 0x148e59f8UL, 0x3c89eb13UL, + 0x27eecea9UL, 0xc935b761UL, 0xe5ede11cUL, 0xb13c7a47UL, + 0xdf599cd2UL, 0x733f55f2UL, 0xce791814UL, 0x37bf73c7UL, + 0xcdea53f7UL, 0xaa5b5ffdUL, 0x6f14df3dUL, 0xdb867844UL, + 0xf381caafUL, 0xc43eb968UL, 0x342c3824UL, 0x405fc2a3UL, + 0xc372161dUL, 0x250cbce2UL, 0x498b283cUL, 0x9541ff0dUL, + 0x017139a8UL, 0xb3de080cUL, 0xe49cd8b4UL, 0xc1906456UL, + 0x84617bcbUL, 0xb670d532UL, 0x5c74486cUL, 0x5742d0b8UL, +}; +static const ulong32 TD3[256] = { + 0xf4a75051UL, 0x4165537eUL, 0x17a4c31aUL, 0x275e963aUL, + 0xab6bcb3bUL, 0x9d45f11fUL, 0xfa58abacUL, 0xe303934bUL, + 0x30fa5520UL, 0x766df6adUL, 0xcc769188UL, 0x024c25f5UL, + 0xe5d7fc4fUL, 0x2acbd7c5UL, 0x35448026UL, 0x62a38fb5UL, + 0xb15a49deUL, 0xba1b6725UL, 0xea0e9845UL, 0xfec0e15dUL, + 0x2f7502c3UL, 0x4cf01281UL, 0x4697a38dUL, 0xd3f9c66bUL, + 0x8f5fe703UL, 0x929c9515UL, 0x6d7aebbfUL, 0x5259da95UL, + 0xbe832dd4UL, 0x7421d358UL, 0xe0692949UL, 0xc9c8448eUL, + 0xc2896a75UL, 0x8e7978f4UL, 0x583e6b99UL, 0xb971dd27UL, + 0xe14fb6beUL, 0x88ad17f0UL, 0x20ac66c9UL, 0xce3ab47dUL, + 0xdf4a1863UL, 0x1a3182e5UL, 0x51336097UL, 0x537f4562UL, + 0x6477e0b1UL, 0x6bae84bbUL, 0x81a01cfeUL, 0x082b94f9UL, + 0x48685870UL, 0x45fd198fUL, 0xde6c8794UL, 0x7bf8b752UL, + 0x73d323abUL, 0x4b02e272UL, 0x1f8f57e3UL, 0x55ab2a66UL, + 0xeb2807b2UL, 0xb5c2032fUL, 0xc57b9a86UL, 0x3708a5d3UL, + 0x2887f230UL, 0xbfa5b223UL, 0x036aba02UL, 0x16825cedUL, + 0xcf1c2b8aUL, 0x79b492a7UL, 0x07f2f0f3UL, 0x69e2a14eUL, + 0xdaf4cd65UL, 0x05bed506UL, 0x34621fd1UL, 0xa6fe8ac4UL, + 0x2e539d34UL, 0xf355a0a2UL, 0x8ae13205UL, 0xf6eb75a4UL, + 0x83ec390bUL, 0x60efaa40UL, 0x719f065eUL, 0x6e1051bdUL, + 0x218af93eUL, 0xdd063d96UL, 0x3e05aeddUL, 0xe6bd464dUL, + 0x548db591UL, 0xc45d0571UL, 0x06d46f04UL, 0x5015ff60UL, + 0x98fb2419UL, 0xbde997d6UL, 0x4043cc89UL, 0xd99e7767UL, + 0xe842bdb0UL, 0x898b8807UL, 0x195b38e7UL, 0xc8eedb79UL, + 0x7c0a47a1UL, 0x420fe97cUL, 0x841ec9f8UL, 0x00000000UL, + 0x80868309UL, 0x2bed4832UL, 0x1170ac1eUL, 0x5a724e6cUL, + 0x0efffbfdUL, 0x8538560fUL, 0xaed51e3dUL, 0x2d392736UL, + 0x0fd9640aUL, 0x5ca62168UL, 0x5b54d19bUL, 0x362e3a24UL, + 0x0a67b10cUL, 0x57e70f93UL, 0xee96d2b4UL, 0x9b919e1bUL, + 0xc0c54f80UL, 0xdc20a261UL, 0x774b695aUL, 0x121a161cUL, + 0x93ba0ae2UL, 0xa02ae5c0UL, 0x22e0433cUL, 0x1b171d12UL, + 0x090d0b0eUL, 0x8bc7adf2UL, 0xb6a8b92dUL, 0x1ea9c814UL, + 0xf1198557UL, 0x75074cafUL, 0x99ddbbeeUL, 0x7f60fda3UL, + 0x01269ff7UL, 0x72f5bc5cUL, 0x663bc544UL, 0xfb7e345bUL, + 0x4329768bUL, 0x23c6dccbUL, 0xedfc68b6UL, 0xe4f163b8UL, + 0x31dccad7UL, 0x63851042UL, 0x97224013UL, 0xc6112084UL, + 0x4a247d85UL, 0xbb3df8d2UL, 0xf93211aeUL, 0x29a16dc7UL, + 0x9e2f4b1dUL, 0xb230f3dcUL, 0x8652ec0dUL, 0xc1e3d077UL, + 0xb3166c2bUL, 0x70b999a9UL, 0x9448fa11UL, 0xe9642247UL, + 0xfc8cc4a8UL, 0xf03f1aa0UL, 0x7d2cd856UL, 0x3390ef22UL, + 0x494ec787UL, 0x38d1c1d9UL, 0xcaa2fe8cUL, 0xd40b3698UL, + 0xf581cfa6UL, 0x7ade28a5UL, 0xb78e26daUL, 0xadbfa43fUL, + 0x3a9de42cUL, 0x78920d50UL, 0x5fcc9b6aUL, 0x7e466254UL, + 0x8d13c2f6UL, 0xd8b8e890UL, 0x39f75e2eUL, 0xc3aff582UL, + 0x5d80be9fUL, 0xd0937c69UL, 0xd52da96fUL, 0x2512b3cfUL, + 0xac993bc8UL, 0x187da710UL, 0x9c636ee8UL, 0x3bbb7bdbUL, + 0x267809cdUL, 0x5918f46eUL, 0x9ab701ecUL, 0x4f9aa883UL, + 0x956e65e6UL, 0xffe67eaaUL, 0xbccf0821UL, 0x15e8e6efUL, + 0xe79bd9baUL, 0x6f36ce4aUL, 0x9f09d4eaUL, 0xb07cd629UL, + 0xa4b2af31UL, 0x3f23312aUL, 0xa59430c6UL, 0xa266c035UL, + 0x4ebc3774UL, 0x82caa6fcUL, 0x90d0b0e0UL, 0xa7d81533UL, + 0x04984af1UL, 0xecdaf741UL, 0xcd500e7fUL, 0x91f62f17UL, + 0x4dd68d76UL, 0xefb04d43UL, 0xaa4d54ccUL, 0x9604dfe4UL, + 0xd1b5e39eUL, 0x6a881b4cUL, 0x2c1fb8c1UL, 0x65517f46UL, + 0x5eea049dUL, 0x8c355d01UL, 0x877473faUL, 0x0b412efbUL, + 0x671d5ab3UL, 0xdbd25292UL, 0x105633e9UL, 0xd647136dUL, + 0xd7618c9aUL, 0xa10c7a37UL, 0xf8148e59UL, 0x133c89ebUL, + 0xa927eeceUL, 0x61c935b7UL, 0x1ce5ede1UL, 0x47b13c7aUL, + 0xd2df599cUL, 0xf2733f55UL, 0x14ce7918UL, 0xc737bf73UL, + 0xf7cdea53UL, 0xfdaa5b5fUL, 0x3d6f14dfUL, 0x44db8678UL, + 0xaff381caUL, 0x68c43eb9UL, 0x24342c38UL, 0xa3405fc2UL, + 0x1dc37216UL, 0xe2250cbcUL, 0x3c498b28UL, 0x0d9541ffUL, + 0xa8017139UL, 0x0cb3de08UL, 0xb4e49cd8UL, 0x56c19064UL, + 0xcb84617bUL, 0x32b670d5UL, 0x6c5c7448UL, 0xb85742d0UL, +}; + +static const ulong32 Tks0[] = { +0x00000000UL, 0x0e090d0bUL, 0x1c121a16UL, 0x121b171dUL, 0x3824342cUL, 0x362d3927UL, 0x24362e3aUL, 0x2a3f2331UL, +0x70486858UL, 0x7e416553UL, 0x6c5a724eUL, 0x62537f45UL, 0x486c5c74UL, 0x4665517fUL, 0x547e4662UL, 0x5a774b69UL, +0xe090d0b0UL, 0xee99ddbbUL, 0xfc82caa6UL, 0xf28bc7adUL, 0xd8b4e49cUL, 0xd6bde997UL, 0xc4a6fe8aUL, 0xcaaff381UL, +0x90d8b8e8UL, 0x9ed1b5e3UL, 0x8ccaa2feUL, 0x82c3aff5UL, 0xa8fc8cc4UL, 0xa6f581cfUL, 0xb4ee96d2UL, 0xbae79bd9UL, +0xdb3bbb7bUL, 0xd532b670UL, 0xc729a16dUL, 0xc920ac66UL, 0xe31f8f57UL, 0xed16825cUL, 0xff0d9541UL, 0xf104984aUL, +0xab73d323UL, 0xa57ade28UL, 0xb761c935UL, 0xb968c43eUL, 0x9357e70fUL, 0x9d5eea04UL, 0x8f45fd19UL, 0x814cf012UL, +0x3bab6bcbUL, 0x35a266c0UL, 0x27b971ddUL, 0x29b07cd6UL, 0x038f5fe7UL, 0x0d8652ecUL, 0x1f9d45f1UL, 0x119448faUL, +0x4be30393UL, 0x45ea0e98UL, 0x57f11985UL, 0x59f8148eUL, 0x73c737bfUL, 0x7dce3ab4UL, 0x6fd52da9UL, 0x61dc20a2UL, +0xad766df6UL, 0xa37f60fdUL, 0xb16477e0UL, 0xbf6d7aebUL, 0x955259daUL, 0x9b5b54d1UL, 0x894043ccUL, 0x87494ec7UL, +0xdd3e05aeUL, 0xd33708a5UL, 0xc12c1fb8UL, 0xcf2512b3UL, 0xe51a3182UL, 0xeb133c89UL, 0xf9082b94UL, 0xf701269fUL, +0x4de6bd46UL, 0x43efb04dUL, 0x51f4a750UL, 0x5ffdaa5bUL, 0x75c2896aUL, 0x7bcb8461UL, 0x69d0937cUL, 0x67d99e77UL, +0x3daed51eUL, 0x33a7d815UL, 0x21bccf08UL, 0x2fb5c203UL, 0x058ae132UL, 0x0b83ec39UL, 0x1998fb24UL, 0x1791f62fUL, +0x764dd68dUL, 0x7844db86UL, 0x6a5fcc9bUL, 0x6456c190UL, 0x4e69e2a1UL, 0x4060efaaUL, 0x527bf8b7UL, 0x5c72f5bcUL, +0x0605bed5UL, 0x080cb3deUL, 0x1a17a4c3UL, 0x141ea9c8UL, 0x3e218af9UL, 0x302887f2UL, 0x223390efUL, 0x2c3a9de4UL, +0x96dd063dUL, 0x98d40b36UL, 0x8acf1c2bUL, 0x84c61120UL, 0xaef93211UL, 0xa0f03f1aUL, 0xb2eb2807UL, 0xbce2250cUL, +0xe6956e65UL, 0xe89c636eUL, 0xfa877473UL, 0xf48e7978UL, 0xdeb15a49UL, 0xd0b85742UL, 0xc2a3405fUL, 0xccaa4d54UL, +0x41ecdaf7UL, 0x4fe5d7fcUL, 0x5dfec0e1UL, 0x53f7cdeaUL, 0x79c8eedbUL, 0x77c1e3d0UL, 0x65daf4cdUL, 0x6bd3f9c6UL, +0x31a4b2afUL, 0x3fadbfa4UL, 0x2db6a8b9UL, 0x23bfa5b2UL, 0x09808683UL, 0x07898b88UL, 0x15929c95UL, 0x1b9b919eUL, +0xa17c0a47UL, 0xaf75074cUL, 0xbd6e1051UL, 0xb3671d5aUL, 0x99583e6bUL, 0x97513360UL, 0x854a247dUL, 0x8b432976UL, +0xd134621fUL, 0xdf3d6f14UL, 0xcd267809UL, 0xc32f7502UL, 0xe9105633UL, 0xe7195b38UL, 0xf5024c25UL, 0xfb0b412eUL, +0x9ad7618cUL, 0x94de6c87UL, 0x86c57b9aUL, 0x88cc7691UL, 0xa2f355a0UL, 0xacfa58abUL, 0xbee14fb6UL, 0xb0e842bdUL, +0xea9f09d4UL, 0xe49604dfUL, 0xf68d13c2UL, 0xf8841ec9UL, 0xd2bb3df8UL, 0xdcb230f3UL, 0xcea927eeUL, 0xc0a02ae5UL, +0x7a47b13cUL, 0x744ebc37UL, 0x6655ab2aUL, 0x685ca621UL, 0x42638510UL, 0x4c6a881bUL, 0x5e719f06UL, 0x5078920dUL, +0x0a0fd964UL, 0x0406d46fUL, 0x161dc372UL, 0x1814ce79UL, 0x322bed48UL, 0x3c22e043UL, 0x2e39f75eUL, 0x2030fa55UL, +0xec9ab701UL, 0xe293ba0aUL, 0xf088ad17UL, 0xfe81a01cUL, 0xd4be832dUL, 0xdab78e26UL, 0xc8ac993bUL, 0xc6a59430UL, +0x9cd2df59UL, 0x92dbd252UL, 0x80c0c54fUL, 0x8ec9c844UL, 0xa4f6eb75UL, 0xaaffe67eUL, 0xb8e4f163UL, 0xb6edfc68UL, +0x0c0a67b1UL, 0x02036abaUL, 0x10187da7UL, 0x1e1170acUL, 0x342e539dUL, 0x3a275e96UL, 0x283c498bUL, 0x26354480UL, +0x7c420fe9UL, 0x724b02e2UL, 0x605015ffUL, 0x6e5918f4UL, 0x44663bc5UL, 0x4a6f36ceUL, 0x587421d3UL, 0x567d2cd8UL, +0x37a10c7aUL, 0x39a80171UL, 0x2bb3166cUL, 0x25ba1b67UL, 0x0f853856UL, 0x018c355dUL, 0x13972240UL, 0x1d9e2f4bUL, +0x47e96422UL, 0x49e06929UL, 0x5bfb7e34UL, 0x55f2733fUL, 0x7fcd500eUL, 0x71c45d05UL, 0x63df4a18UL, 0x6dd64713UL, +0xd731dccaUL, 0xd938d1c1UL, 0xcb23c6dcUL, 0xc52acbd7UL, 0xef15e8e6UL, 0xe11ce5edUL, 0xf307f2f0UL, 0xfd0efffbUL, +0xa779b492UL, 0xa970b999UL, 0xbb6bae84UL, 0xb562a38fUL, 0x9f5d80beUL, 0x91548db5UL, 0x834f9aa8UL, 0x8d4697a3UL +}; + +static const ulong32 Tks1[] = { +0x00000000UL, 0x0b0e090dUL, 0x161c121aUL, 0x1d121b17UL, 0x2c382434UL, 0x27362d39UL, 0x3a24362eUL, 0x312a3f23UL, +0x58704868UL, 0x537e4165UL, 0x4e6c5a72UL, 0x4562537fUL, 0x74486c5cUL, 0x7f466551UL, 0x62547e46UL, 0x695a774bUL, +0xb0e090d0UL, 0xbbee99ddUL, 0xa6fc82caUL, 0xadf28bc7UL, 0x9cd8b4e4UL, 0x97d6bde9UL, 0x8ac4a6feUL, 0x81caaff3UL, +0xe890d8b8UL, 0xe39ed1b5UL, 0xfe8ccaa2UL, 0xf582c3afUL, 0xc4a8fc8cUL, 0xcfa6f581UL, 0xd2b4ee96UL, 0xd9bae79bUL, +0x7bdb3bbbUL, 0x70d532b6UL, 0x6dc729a1UL, 0x66c920acUL, 0x57e31f8fUL, 0x5ced1682UL, 0x41ff0d95UL, 0x4af10498UL, +0x23ab73d3UL, 0x28a57adeUL, 0x35b761c9UL, 0x3eb968c4UL, 0x0f9357e7UL, 0x049d5eeaUL, 0x198f45fdUL, 0x12814cf0UL, +0xcb3bab6bUL, 0xc035a266UL, 0xdd27b971UL, 0xd629b07cUL, 0xe7038f5fUL, 0xec0d8652UL, 0xf11f9d45UL, 0xfa119448UL, +0x934be303UL, 0x9845ea0eUL, 0x8557f119UL, 0x8e59f814UL, 0xbf73c737UL, 0xb47dce3aUL, 0xa96fd52dUL, 0xa261dc20UL, +0xf6ad766dUL, 0xfda37f60UL, 0xe0b16477UL, 0xebbf6d7aUL, 0xda955259UL, 0xd19b5b54UL, 0xcc894043UL, 0xc787494eUL, +0xaedd3e05UL, 0xa5d33708UL, 0xb8c12c1fUL, 0xb3cf2512UL, 0x82e51a31UL, 0x89eb133cUL, 0x94f9082bUL, 0x9ff70126UL, +0x464de6bdUL, 0x4d43efb0UL, 0x5051f4a7UL, 0x5b5ffdaaUL, 0x6a75c289UL, 0x617bcb84UL, 0x7c69d093UL, 0x7767d99eUL, +0x1e3daed5UL, 0x1533a7d8UL, 0x0821bccfUL, 0x032fb5c2UL, 0x32058ae1UL, 0x390b83ecUL, 0x241998fbUL, 0x2f1791f6UL, +0x8d764dd6UL, 0x867844dbUL, 0x9b6a5fccUL, 0x906456c1UL, 0xa14e69e2UL, 0xaa4060efUL, 0xb7527bf8UL, 0xbc5c72f5UL, +0xd50605beUL, 0xde080cb3UL, 0xc31a17a4UL, 0xc8141ea9UL, 0xf93e218aUL, 0xf2302887UL, 0xef223390UL, 0xe42c3a9dUL, +0x3d96dd06UL, 0x3698d40bUL, 0x2b8acf1cUL, 0x2084c611UL, 0x11aef932UL, 0x1aa0f03fUL, 0x07b2eb28UL, 0x0cbce225UL, +0x65e6956eUL, 0x6ee89c63UL, 0x73fa8774UL, 0x78f48e79UL, 0x49deb15aUL, 0x42d0b857UL, 0x5fc2a340UL, 0x54ccaa4dUL, +0xf741ecdaUL, 0xfc4fe5d7UL, 0xe15dfec0UL, 0xea53f7cdUL, 0xdb79c8eeUL, 0xd077c1e3UL, 0xcd65daf4UL, 0xc66bd3f9UL, +0xaf31a4b2UL, 0xa43fadbfUL, 0xb92db6a8UL, 0xb223bfa5UL, 0x83098086UL, 0x8807898bUL, 0x9515929cUL, 0x9e1b9b91UL, +0x47a17c0aUL, 0x4caf7507UL, 0x51bd6e10UL, 0x5ab3671dUL, 0x6b99583eUL, 0x60975133UL, 0x7d854a24UL, 0x768b4329UL, +0x1fd13462UL, 0x14df3d6fUL, 0x09cd2678UL, 0x02c32f75UL, 0x33e91056UL, 0x38e7195bUL, 0x25f5024cUL, 0x2efb0b41UL, +0x8c9ad761UL, 0x8794de6cUL, 0x9a86c57bUL, 0x9188cc76UL, 0xa0a2f355UL, 0xabacfa58UL, 0xb6bee14fUL, 0xbdb0e842UL, +0xd4ea9f09UL, 0xdfe49604UL, 0xc2f68d13UL, 0xc9f8841eUL, 0xf8d2bb3dUL, 0xf3dcb230UL, 0xeecea927UL, 0xe5c0a02aUL, +0x3c7a47b1UL, 0x37744ebcUL, 0x2a6655abUL, 0x21685ca6UL, 0x10426385UL, 0x1b4c6a88UL, 0x065e719fUL, 0x0d507892UL, +0x640a0fd9UL, 0x6f0406d4UL, 0x72161dc3UL, 0x791814ceUL, 0x48322bedUL, 0x433c22e0UL, 0x5e2e39f7UL, 0x552030faUL, +0x01ec9ab7UL, 0x0ae293baUL, 0x17f088adUL, 0x1cfe81a0UL, 0x2dd4be83UL, 0x26dab78eUL, 0x3bc8ac99UL, 0x30c6a594UL, +0x599cd2dfUL, 0x5292dbd2UL, 0x4f80c0c5UL, 0x448ec9c8UL, 0x75a4f6ebUL, 0x7eaaffe6UL, 0x63b8e4f1UL, 0x68b6edfcUL, +0xb10c0a67UL, 0xba02036aUL, 0xa710187dUL, 0xac1e1170UL, 0x9d342e53UL, 0x963a275eUL, 0x8b283c49UL, 0x80263544UL, +0xe97c420fUL, 0xe2724b02UL, 0xff605015UL, 0xf46e5918UL, 0xc544663bUL, 0xce4a6f36UL, 0xd3587421UL, 0xd8567d2cUL, +0x7a37a10cUL, 0x7139a801UL, 0x6c2bb316UL, 0x6725ba1bUL, 0x560f8538UL, 0x5d018c35UL, 0x40139722UL, 0x4b1d9e2fUL, +0x2247e964UL, 0x2949e069UL, 0x345bfb7eUL, 0x3f55f273UL, 0x0e7fcd50UL, 0x0571c45dUL, 0x1863df4aUL, 0x136dd647UL, +0xcad731dcUL, 0xc1d938d1UL, 0xdccb23c6UL, 0xd7c52acbUL, 0xe6ef15e8UL, 0xede11ce5UL, 0xf0f307f2UL, 0xfbfd0effUL, +0x92a779b4UL, 0x99a970b9UL, 0x84bb6baeUL, 0x8fb562a3UL, 0xbe9f5d80UL, 0xb591548dUL, 0xa8834f9aUL, 0xa38d4697UL +}; + +static const ulong32 Tks2[] = { +0x00000000UL, 0x0d0b0e09UL, 0x1a161c12UL, 0x171d121bUL, 0x342c3824UL, 0x3927362dUL, 0x2e3a2436UL, 0x23312a3fUL, +0x68587048UL, 0x65537e41UL, 0x724e6c5aUL, 0x7f456253UL, 0x5c74486cUL, 0x517f4665UL, 0x4662547eUL, 0x4b695a77UL, +0xd0b0e090UL, 0xddbbee99UL, 0xcaa6fc82UL, 0xc7adf28bUL, 0xe49cd8b4UL, 0xe997d6bdUL, 0xfe8ac4a6UL, 0xf381caafUL, +0xb8e890d8UL, 0xb5e39ed1UL, 0xa2fe8ccaUL, 0xaff582c3UL, 0x8cc4a8fcUL, 0x81cfa6f5UL, 0x96d2b4eeUL, 0x9bd9bae7UL, +0xbb7bdb3bUL, 0xb670d532UL, 0xa16dc729UL, 0xac66c920UL, 0x8f57e31fUL, 0x825ced16UL, 0x9541ff0dUL, 0x984af104UL, +0xd323ab73UL, 0xde28a57aUL, 0xc935b761UL, 0xc43eb968UL, 0xe70f9357UL, 0xea049d5eUL, 0xfd198f45UL, 0xf012814cUL, +0x6bcb3babUL, 0x66c035a2UL, 0x71dd27b9UL, 0x7cd629b0UL, 0x5fe7038fUL, 0x52ec0d86UL, 0x45f11f9dUL, 0x48fa1194UL, +0x03934be3UL, 0x0e9845eaUL, 0x198557f1UL, 0x148e59f8UL, 0x37bf73c7UL, 0x3ab47dceUL, 0x2da96fd5UL, 0x20a261dcUL, +0x6df6ad76UL, 0x60fda37fUL, 0x77e0b164UL, 0x7aebbf6dUL, 0x59da9552UL, 0x54d19b5bUL, 0x43cc8940UL, 0x4ec78749UL, +0x05aedd3eUL, 0x08a5d337UL, 0x1fb8c12cUL, 0x12b3cf25UL, 0x3182e51aUL, 0x3c89eb13UL, 0x2b94f908UL, 0x269ff701UL, +0xbd464de6UL, 0xb04d43efUL, 0xa75051f4UL, 0xaa5b5ffdUL, 0x896a75c2UL, 0x84617bcbUL, 0x937c69d0UL, 0x9e7767d9UL, +0xd51e3daeUL, 0xd81533a7UL, 0xcf0821bcUL, 0xc2032fb5UL, 0xe132058aUL, 0xec390b83UL, 0xfb241998UL, 0xf62f1791UL, +0xd68d764dUL, 0xdb867844UL, 0xcc9b6a5fUL, 0xc1906456UL, 0xe2a14e69UL, 0xefaa4060UL, 0xf8b7527bUL, 0xf5bc5c72UL, +0xbed50605UL, 0xb3de080cUL, 0xa4c31a17UL, 0xa9c8141eUL, 0x8af93e21UL, 0x87f23028UL, 0x90ef2233UL, 0x9de42c3aUL, +0x063d96ddUL, 0x0b3698d4UL, 0x1c2b8acfUL, 0x112084c6UL, 0x3211aef9UL, 0x3f1aa0f0UL, 0x2807b2ebUL, 0x250cbce2UL, +0x6e65e695UL, 0x636ee89cUL, 0x7473fa87UL, 0x7978f48eUL, 0x5a49deb1UL, 0x5742d0b8UL, 0x405fc2a3UL, 0x4d54ccaaUL, +0xdaf741ecUL, 0xd7fc4fe5UL, 0xc0e15dfeUL, 0xcdea53f7UL, 0xeedb79c8UL, 0xe3d077c1UL, 0xf4cd65daUL, 0xf9c66bd3UL, +0xb2af31a4UL, 0xbfa43fadUL, 0xa8b92db6UL, 0xa5b223bfUL, 0x86830980UL, 0x8b880789UL, 0x9c951592UL, 0x919e1b9bUL, +0x0a47a17cUL, 0x074caf75UL, 0x1051bd6eUL, 0x1d5ab367UL, 0x3e6b9958UL, 0x33609751UL, 0x247d854aUL, 0x29768b43UL, +0x621fd134UL, 0x6f14df3dUL, 0x7809cd26UL, 0x7502c32fUL, 0x5633e910UL, 0x5b38e719UL, 0x4c25f502UL, 0x412efb0bUL, +0x618c9ad7UL, 0x6c8794deUL, 0x7b9a86c5UL, 0x769188ccUL, 0x55a0a2f3UL, 0x58abacfaUL, 0x4fb6bee1UL, 0x42bdb0e8UL, +0x09d4ea9fUL, 0x04dfe496UL, 0x13c2f68dUL, 0x1ec9f884UL, 0x3df8d2bbUL, 0x30f3dcb2UL, 0x27eecea9UL, 0x2ae5c0a0UL, +0xb13c7a47UL, 0xbc37744eUL, 0xab2a6655UL, 0xa621685cUL, 0x85104263UL, 0x881b4c6aUL, 0x9f065e71UL, 0x920d5078UL, +0xd9640a0fUL, 0xd46f0406UL, 0xc372161dUL, 0xce791814UL, 0xed48322bUL, 0xe0433c22UL, 0xf75e2e39UL, 0xfa552030UL, +0xb701ec9aUL, 0xba0ae293UL, 0xad17f088UL, 0xa01cfe81UL, 0x832dd4beUL, 0x8e26dab7UL, 0x993bc8acUL, 0x9430c6a5UL, +0xdf599cd2UL, 0xd25292dbUL, 0xc54f80c0UL, 0xc8448ec9UL, 0xeb75a4f6UL, 0xe67eaaffUL, 0xf163b8e4UL, 0xfc68b6edUL, +0x67b10c0aUL, 0x6aba0203UL, 0x7da71018UL, 0x70ac1e11UL, 0x539d342eUL, 0x5e963a27UL, 0x498b283cUL, 0x44802635UL, +0x0fe97c42UL, 0x02e2724bUL, 0x15ff6050UL, 0x18f46e59UL, 0x3bc54466UL, 0x36ce4a6fUL, 0x21d35874UL, 0x2cd8567dUL, +0x0c7a37a1UL, 0x017139a8UL, 0x166c2bb3UL, 0x1b6725baUL, 0x38560f85UL, 0x355d018cUL, 0x22401397UL, 0x2f4b1d9eUL, +0x642247e9UL, 0x692949e0UL, 0x7e345bfbUL, 0x733f55f2UL, 0x500e7fcdUL, 0x5d0571c4UL, 0x4a1863dfUL, 0x47136dd6UL, +0xdccad731UL, 0xd1c1d938UL, 0xc6dccb23UL, 0xcbd7c52aUL, 0xe8e6ef15UL, 0xe5ede11cUL, 0xf2f0f307UL, 0xfffbfd0eUL, +0xb492a779UL, 0xb999a970UL, 0xae84bb6bUL, 0xa38fb562UL, 0x80be9f5dUL, 0x8db59154UL, 0x9aa8834fUL, 0x97a38d46UL +}; + +static const ulong32 Tks3[] = { +0x00000000UL, 0x090d0b0eUL, 0x121a161cUL, 0x1b171d12UL, 0x24342c38UL, 0x2d392736UL, 0x362e3a24UL, 0x3f23312aUL, +0x48685870UL, 0x4165537eUL, 0x5a724e6cUL, 0x537f4562UL, 0x6c5c7448UL, 0x65517f46UL, 0x7e466254UL, 0x774b695aUL, +0x90d0b0e0UL, 0x99ddbbeeUL, 0x82caa6fcUL, 0x8bc7adf2UL, 0xb4e49cd8UL, 0xbde997d6UL, 0xa6fe8ac4UL, 0xaff381caUL, +0xd8b8e890UL, 0xd1b5e39eUL, 0xcaa2fe8cUL, 0xc3aff582UL, 0xfc8cc4a8UL, 0xf581cfa6UL, 0xee96d2b4UL, 0xe79bd9baUL, +0x3bbb7bdbUL, 0x32b670d5UL, 0x29a16dc7UL, 0x20ac66c9UL, 0x1f8f57e3UL, 0x16825cedUL, 0x0d9541ffUL, 0x04984af1UL, +0x73d323abUL, 0x7ade28a5UL, 0x61c935b7UL, 0x68c43eb9UL, 0x57e70f93UL, 0x5eea049dUL, 0x45fd198fUL, 0x4cf01281UL, +0xab6bcb3bUL, 0xa266c035UL, 0xb971dd27UL, 0xb07cd629UL, 0x8f5fe703UL, 0x8652ec0dUL, 0x9d45f11fUL, 0x9448fa11UL, +0xe303934bUL, 0xea0e9845UL, 0xf1198557UL, 0xf8148e59UL, 0xc737bf73UL, 0xce3ab47dUL, 0xd52da96fUL, 0xdc20a261UL, +0x766df6adUL, 0x7f60fda3UL, 0x6477e0b1UL, 0x6d7aebbfUL, 0x5259da95UL, 0x5b54d19bUL, 0x4043cc89UL, 0x494ec787UL, +0x3e05aeddUL, 0x3708a5d3UL, 0x2c1fb8c1UL, 0x2512b3cfUL, 0x1a3182e5UL, 0x133c89ebUL, 0x082b94f9UL, 0x01269ff7UL, +0xe6bd464dUL, 0xefb04d43UL, 0xf4a75051UL, 0xfdaa5b5fUL, 0xc2896a75UL, 0xcb84617bUL, 0xd0937c69UL, 0xd99e7767UL, +0xaed51e3dUL, 0xa7d81533UL, 0xbccf0821UL, 0xb5c2032fUL, 0x8ae13205UL, 0x83ec390bUL, 0x98fb2419UL, 0x91f62f17UL, +0x4dd68d76UL, 0x44db8678UL, 0x5fcc9b6aUL, 0x56c19064UL, 0x69e2a14eUL, 0x60efaa40UL, 0x7bf8b752UL, 0x72f5bc5cUL, +0x05bed506UL, 0x0cb3de08UL, 0x17a4c31aUL, 0x1ea9c814UL, 0x218af93eUL, 0x2887f230UL, 0x3390ef22UL, 0x3a9de42cUL, +0xdd063d96UL, 0xd40b3698UL, 0xcf1c2b8aUL, 0xc6112084UL, 0xf93211aeUL, 0xf03f1aa0UL, 0xeb2807b2UL, 0xe2250cbcUL, +0x956e65e6UL, 0x9c636ee8UL, 0x877473faUL, 0x8e7978f4UL, 0xb15a49deUL, 0xb85742d0UL, 0xa3405fc2UL, 0xaa4d54ccUL, +0xecdaf741UL, 0xe5d7fc4fUL, 0xfec0e15dUL, 0xf7cdea53UL, 0xc8eedb79UL, 0xc1e3d077UL, 0xdaf4cd65UL, 0xd3f9c66bUL, +0xa4b2af31UL, 0xadbfa43fUL, 0xb6a8b92dUL, 0xbfa5b223UL, 0x80868309UL, 0x898b8807UL, 0x929c9515UL, 0x9b919e1bUL, +0x7c0a47a1UL, 0x75074cafUL, 0x6e1051bdUL, 0x671d5ab3UL, 0x583e6b99UL, 0x51336097UL, 0x4a247d85UL, 0x4329768bUL, +0x34621fd1UL, 0x3d6f14dfUL, 0x267809cdUL, 0x2f7502c3UL, 0x105633e9UL, 0x195b38e7UL, 0x024c25f5UL, 0x0b412efbUL, +0xd7618c9aUL, 0xde6c8794UL, 0xc57b9a86UL, 0xcc769188UL, 0xf355a0a2UL, 0xfa58abacUL, 0xe14fb6beUL, 0xe842bdb0UL, +0x9f09d4eaUL, 0x9604dfe4UL, 0x8d13c2f6UL, 0x841ec9f8UL, 0xbb3df8d2UL, 0xb230f3dcUL, 0xa927eeceUL, 0xa02ae5c0UL, +0x47b13c7aUL, 0x4ebc3774UL, 0x55ab2a66UL, 0x5ca62168UL, 0x63851042UL, 0x6a881b4cUL, 0x719f065eUL, 0x78920d50UL, +0x0fd9640aUL, 0x06d46f04UL, 0x1dc37216UL, 0x14ce7918UL, 0x2bed4832UL, 0x22e0433cUL, 0x39f75e2eUL, 0x30fa5520UL, +0x9ab701ecUL, 0x93ba0ae2UL, 0x88ad17f0UL, 0x81a01cfeUL, 0xbe832dd4UL, 0xb78e26daUL, 0xac993bc8UL, 0xa59430c6UL, +0xd2df599cUL, 0xdbd25292UL, 0xc0c54f80UL, 0xc9c8448eUL, 0xf6eb75a4UL, 0xffe67eaaUL, 0xe4f163b8UL, 0xedfc68b6UL, +0x0a67b10cUL, 0x036aba02UL, 0x187da710UL, 0x1170ac1eUL, 0x2e539d34UL, 0x275e963aUL, 0x3c498b28UL, 0x35448026UL, +0x420fe97cUL, 0x4b02e272UL, 0x5015ff60UL, 0x5918f46eUL, 0x663bc544UL, 0x6f36ce4aUL, 0x7421d358UL, 0x7d2cd856UL, +0xa10c7a37UL, 0xa8017139UL, 0xb3166c2bUL, 0xba1b6725UL, 0x8538560fUL, 0x8c355d01UL, 0x97224013UL, 0x9e2f4b1dUL, +0xe9642247UL, 0xe0692949UL, 0xfb7e345bUL, 0xf2733f55UL, 0xcd500e7fUL, 0xc45d0571UL, 0xdf4a1863UL, 0xd647136dUL, +0x31dccad7UL, 0x38d1c1d9UL, 0x23c6dccbUL, 0x2acbd7c5UL, 0x15e8e6efUL, 0x1ce5ede1UL, 0x07f2f0f3UL, 0x0efffbfdUL, +0x79b492a7UL, 0x70b999a9UL, 0x6bae84bbUL, 0x62a38fb5UL, 0x5d80be9fUL, 0x548db591UL, 0x4f9aa883UL, 0x4697a38dUL +}; + +#endif /* ENCRYPT_ONLY */ + +#endif /* SMALL CODE */ + +static const ulong32 rcon[] = { + 0x01000000UL, 0x02000000UL, 0x04000000UL, 0x08000000UL, + 0x10000000UL, 0x20000000UL, 0x40000000UL, 0x80000000UL, + 0x1B000000UL, 0x36000000UL, /* for 128-bit blocks, Rijndael never uses more than 10 rcon values */ +}; + +#endif /* __LTC_AES_TAB_C__ */ + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/* AES implementation by Tom St Denis + * + * Derived from the Public Domain source code by + +--- + * rijndael-alg-fst.c + * + * @version 3.0 (December 2000) + * + * Optimised ANSI C code for the Rijndael cipher (now AES) + * + * @author Vincent Rijmen + * @author Antoon Bosselaers + * @author Paulo Barreto +--- + */ +/** + @file aes.c + Implementation of AES +*/ + + + +#ifdef LTC_RIJNDAEL + +#ifndef ENCRYPT_ONLY + +#define SETUP rijndael_setup +#define ECB_ENC rijndael_ecb_encrypt +#define ECB_DEC rijndael_ecb_decrypt +#define ECB_DONE rijndael_done +#define ECB_TEST rijndael_test +#define ECB_KS rijndael_keysize + +const struct ltc_cipher_descriptor rijndael_desc = +{ + "rijndael", + 6, + 16, 32, 16, 10, + SETUP, ECB_ENC, ECB_DEC, ECB_TEST, ECB_DONE, ECB_KS, + NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL +}; + +const struct ltc_cipher_descriptor aes_desc = +{ + "aes", + 6, + 16, 32, 16, 10, + SETUP, ECB_ENC, ECB_DEC, ECB_TEST, ECB_DONE, ECB_KS, + NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL +}; + +#else + +#define SETUP rijndael_enc_setup +#define ECB_ENC rijndael_enc_ecb_encrypt +#define ECB_KS rijndael_enc_keysize +#define ECB_DONE rijndael_enc_done + +const struct ltc_cipher_descriptor rijndael_enc_desc = +{ + "rijndael", + 6, + 16, 32, 16, 10, + SETUP, ECB_ENC, NULL, NULL, ECB_DONE, ECB_KS, + NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL +}; + +const struct ltc_cipher_descriptor aes_enc_desc = +{ + "aes", + 6, + 16, 32, 16, 10, + SETUP, ECB_ENC, NULL, NULL, ECB_DONE, ECB_KS, + NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL +}; + +#endif + +#define __LTC_AES_TAB_C__ + + +static ulong32 setup_mix(ulong32 temp) +{ + return (Te4_3[byte(temp, 2)]) ^ + (Te4_2[byte(temp, 1)]) ^ + (Te4_1[byte(temp, 0)]) ^ + (Te4_0[byte(temp, 3)]); +} + +#ifndef ENCRYPT_ONLY +#ifdef LTC_SMALL_CODE +static ulong32 setup_mix2(ulong32 temp) +{ + return Td0(255 & Te4[byte(temp, 3)]) ^ + Td1(255 & Te4[byte(temp, 2)]) ^ + Td2(255 & Te4[byte(temp, 1)]) ^ + Td3(255 & Te4[byte(temp, 0)]); +} +#endif +#endif + + /** + Initialize the AES (Rijndael) block cipher + @param key The symmetric key you wish to pass + @param keylen The key length in bytes + @param num_rounds The number of rounds desired (0 for default) + @param skey The key in as scheduled by this function. + @return CRYPT_OK if successful + */ +int SETUP(const unsigned char *key, int keylen, int num_rounds, symmetric_key *skey) +{ + int i; + ulong32 temp, *rk; +#ifndef ENCRYPT_ONLY + ulong32 *rrk; +#endif + LTC_ARGCHK(key != NULL); + LTC_ARGCHK(skey != NULL); + + if (keylen != 16 && keylen != 24 && keylen != 32) { + return CRYPT_INVALID_KEYSIZE; + } + + if (num_rounds != 0 && num_rounds != (10 + ((keylen/8)-2)*2)) { + return CRYPT_INVALID_ROUNDS; + } + + skey->rijndael.Nr = 10 + ((keylen/8)-2)*2; + + /* setup the forward key */ + i = 0; + rk = skey->rijndael.eK; + LOAD32H(rk[0], key ); + LOAD32H(rk[1], key + 4); + LOAD32H(rk[2], key + 8); + LOAD32H(rk[3], key + 12); + if (keylen == 16) { + for (;;) { + temp = rk[3]; + rk[4] = rk[0] ^ setup_mix(temp) ^ rcon[i]; + rk[5] = rk[1] ^ rk[4]; + rk[6] = rk[2] ^ rk[5]; + rk[7] = rk[3] ^ rk[6]; + if (++i == 10) { + break; + } + rk += 4; + } + } else if (keylen == 24) { + LOAD32H(rk[4], key + 16); + LOAD32H(rk[5], key + 20); + for (;;) { + #ifdef _MSC_VER + temp = skey->rijndael.eK[rk - skey->rijndael.eK + 5]; + #else + temp = rk[5]; + #endif + rk[ 6] = rk[ 0] ^ setup_mix(temp) ^ rcon[i]; + rk[ 7] = rk[ 1] ^ rk[ 6]; + rk[ 8] = rk[ 2] ^ rk[ 7]; + rk[ 9] = rk[ 3] ^ rk[ 8]; + if (++i == 8) { + break; + } + rk[10] = rk[ 4] ^ rk[ 9]; + rk[11] = rk[ 5] ^ rk[10]; + rk += 6; + } + } else if (keylen == 32) { + LOAD32H(rk[4], key + 16); + LOAD32H(rk[5], key + 20); + LOAD32H(rk[6], key + 24); + LOAD32H(rk[7], key + 28); + for (;;) { + #ifdef _MSC_VER + temp = skey->rijndael.eK[rk - skey->rijndael.eK + 7]; + #else + temp = rk[7]; + #endif + rk[ 8] = rk[ 0] ^ setup_mix(temp) ^ rcon[i]; + rk[ 9] = rk[ 1] ^ rk[ 8]; + rk[10] = rk[ 2] ^ rk[ 9]; + rk[11] = rk[ 3] ^ rk[10]; + if (++i == 7) { + break; + } + temp = rk[11]; + rk[12] = rk[ 4] ^ setup_mix(RORc(temp, 8)); + rk[13] = rk[ 5] ^ rk[12]; + rk[14] = rk[ 6] ^ rk[13]; + rk[15] = rk[ 7] ^ rk[14]; + rk += 8; + } + } else { + /* this can't happen */ + /* coverity[dead_error_line] */ + return CRYPT_ERROR; + } + +#ifndef ENCRYPT_ONLY + /* setup the inverse key now */ + rk = skey->rijndael.dK; + rrk = skey->rijndael.eK + (28 + keylen) - 4; + + /* apply the inverse MixColumn transform to all round keys but the first and the last: */ + /* copy first */ + *rk++ = *rrk++; + *rk++ = *rrk++; + *rk++ = *rrk++; + *rk = *rrk; + rk -= 3; rrk -= 3; + + for (i = 1; i < skey->rijndael.Nr; i++) { + rrk -= 4; + rk += 4; + #ifdef LTC_SMALL_CODE + temp = rrk[0]; + rk[0] = setup_mix2(temp); + temp = rrk[1]; + rk[1] = setup_mix2(temp); + temp = rrk[2]; + rk[2] = setup_mix2(temp); + temp = rrk[3]; + rk[3] = setup_mix2(temp); + #else + temp = rrk[0]; + rk[0] = + Tks0[byte(temp, 3)] ^ + Tks1[byte(temp, 2)] ^ + Tks2[byte(temp, 1)] ^ + Tks3[byte(temp, 0)]; + temp = rrk[1]; + rk[1] = + Tks0[byte(temp, 3)] ^ + Tks1[byte(temp, 2)] ^ + Tks2[byte(temp, 1)] ^ + Tks3[byte(temp, 0)]; + temp = rrk[2]; + rk[2] = + Tks0[byte(temp, 3)] ^ + Tks1[byte(temp, 2)] ^ + Tks2[byte(temp, 1)] ^ + Tks3[byte(temp, 0)]; + temp = rrk[3]; + rk[3] = + Tks0[byte(temp, 3)] ^ + Tks1[byte(temp, 2)] ^ + Tks2[byte(temp, 1)] ^ + Tks3[byte(temp, 0)]; + #endif + + } + + /* copy last */ + rrk -= 4; + rk += 4; + *rk++ = *rrk++; + *rk++ = *rrk++; + *rk++ = *rrk++; + *rk = *rrk; +#endif /* ENCRYPT_ONLY */ + + return CRYPT_OK; +} + +/** + Encrypts a block of text with AES + @param pt The input plaintext (16 bytes) + @param ct The output ciphertext (16 bytes) + @param skey The key as scheduled + @return CRYPT_OK if successful +*/ +#ifdef LTC_CLEAN_STACK +static int _rijndael_ecb_encrypt(const unsigned char *pt, unsigned char *ct, symmetric_key *skey) +#else +int ECB_ENC(const unsigned char *pt, unsigned char *ct, symmetric_key *skey) +#endif +{ + ulong32 s0, s1, s2, s3, t0, t1, t2, t3, *rk; + int Nr, r; + + LTC_ARGCHK(pt != NULL); + LTC_ARGCHK(ct != NULL); + LTC_ARGCHK(skey != NULL); + + Nr = skey->rijndael.Nr; + rk = skey->rijndael.eK; + + /* + * map byte array block to cipher state + * and add initial round key: + */ + LOAD32H(s0, pt ); s0 ^= rk[0]; + LOAD32H(s1, pt + 4); s1 ^= rk[1]; + LOAD32H(s2, pt + 8); s2 ^= rk[2]; + LOAD32H(s3, pt + 12); s3 ^= rk[3]; + +#ifdef LTC_SMALL_CODE + + for (r = 0; ; r++) { + rk += 4; + t0 = + Te0(byte(s0, 3)) ^ + Te1(byte(s1, 2)) ^ + Te2(byte(s2, 1)) ^ + Te3(byte(s3, 0)) ^ + rk[0]; + t1 = + Te0(byte(s1, 3)) ^ + Te1(byte(s2, 2)) ^ + Te2(byte(s3, 1)) ^ + Te3(byte(s0, 0)) ^ + rk[1]; + t2 = + Te0(byte(s2, 3)) ^ + Te1(byte(s3, 2)) ^ + Te2(byte(s0, 1)) ^ + Te3(byte(s1, 0)) ^ + rk[2]; + t3 = + Te0(byte(s3, 3)) ^ + Te1(byte(s0, 2)) ^ + Te2(byte(s1, 1)) ^ + Te3(byte(s2, 0)) ^ + rk[3]; + if (r == Nr-2) { + break; + } + s0 = t0; s1 = t1; s2 = t2; s3 = t3; + } + rk += 4; + +#else + + /* + * Nr - 1 full rounds: + */ + r = Nr >> 1; + for (;;) { + t0 = + Te0(byte(s0, 3)) ^ + Te1(byte(s1, 2)) ^ + Te2(byte(s2, 1)) ^ + Te3(byte(s3, 0)) ^ + rk[4]; + t1 = + Te0(byte(s1, 3)) ^ + Te1(byte(s2, 2)) ^ + Te2(byte(s3, 1)) ^ + Te3(byte(s0, 0)) ^ + rk[5]; + t2 = + Te0(byte(s2, 3)) ^ + Te1(byte(s3, 2)) ^ + Te2(byte(s0, 1)) ^ + Te3(byte(s1, 0)) ^ + rk[6]; + t3 = + Te0(byte(s3, 3)) ^ + Te1(byte(s0, 2)) ^ + Te2(byte(s1, 1)) ^ + Te3(byte(s2, 0)) ^ + rk[7]; + + rk += 8; + if (--r == 0) { + break; + } + + s0 = + Te0(byte(t0, 3)) ^ + Te1(byte(t1, 2)) ^ + Te2(byte(t2, 1)) ^ + Te3(byte(t3, 0)) ^ + rk[0]; + s1 = + Te0(byte(t1, 3)) ^ + Te1(byte(t2, 2)) ^ + Te2(byte(t3, 1)) ^ + Te3(byte(t0, 0)) ^ + rk[1]; + s2 = + Te0(byte(t2, 3)) ^ + Te1(byte(t3, 2)) ^ + Te2(byte(t0, 1)) ^ + Te3(byte(t1, 0)) ^ + rk[2]; + s3 = + Te0(byte(t3, 3)) ^ + Te1(byte(t0, 2)) ^ + Te2(byte(t1, 1)) ^ + Te3(byte(t2, 0)) ^ + rk[3]; + } + +#endif + + /* + * apply last round and + * map cipher state to byte array block: + */ + s0 = + (Te4_3[byte(t0, 3)]) ^ + (Te4_2[byte(t1, 2)]) ^ + (Te4_1[byte(t2, 1)]) ^ + (Te4_0[byte(t3, 0)]) ^ + rk[0]; + STORE32H(s0, ct); + s1 = + (Te4_3[byte(t1, 3)]) ^ + (Te4_2[byte(t2, 2)]) ^ + (Te4_1[byte(t3, 1)]) ^ + (Te4_0[byte(t0, 0)]) ^ + rk[1]; + STORE32H(s1, ct+4); + s2 = + (Te4_3[byte(t2, 3)]) ^ + (Te4_2[byte(t3, 2)]) ^ + (Te4_1[byte(t0, 1)]) ^ + (Te4_0[byte(t1, 0)]) ^ + rk[2]; + STORE32H(s2, ct+8); + s3 = + (Te4_3[byte(t3, 3)]) ^ + (Te4_2[byte(t0, 2)]) ^ + (Te4_1[byte(t1, 1)]) ^ + (Te4_0[byte(t2, 0)]) ^ + rk[3]; + STORE32H(s3, ct+12); + + return CRYPT_OK; +} + +#ifdef LTC_CLEAN_STACK +int ECB_ENC(const unsigned char *pt, unsigned char *ct, symmetric_key *skey) +{ + int err = _rijndael_ecb_encrypt(pt, ct, skey); + burn_stack(sizeof(unsigned long)*8 + sizeof(unsigned long*) + sizeof(int)*2); + return err; +} +#endif + +#ifndef ENCRYPT_ONLY + +/** + Decrypts a block of text with AES + @param ct The input ciphertext (16 bytes) + @param pt The output plaintext (16 bytes) + @param skey The key as scheduled + @return CRYPT_OK if successful +*/ +#ifdef LTC_CLEAN_STACK +static int _rijndael_ecb_decrypt(const unsigned char *ct, unsigned char *pt, symmetric_key *skey) +#else +int ECB_DEC(const unsigned char *ct, unsigned char *pt, symmetric_key *skey) +#endif +{ + ulong32 s0, s1, s2, s3, t0, t1, t2, t3, *rk; + int Nr, r; + + LTC_ARGCHK(pt != NULL); + LTC_ARGCHK(ct != NULL); + LTC_ARGCHK(skey != NULL); + + Nr = skey->rijndael.Nr; + rk = skey->rijndael.dK; + + /* + * map byte array block to cipher state + * and add initial round key: + */ + LOAD32H(s0, ct ); s0 ^= rk[0]; + LOAD32H(s1, ct + 4); s1 ^= rk[1]; + LOAD32H(s2, ct + 8); s2 ^= rk[2]; + LOAD32H(s3, ct + 12); s3 ^= rk[3]; + +#ifdef LTC_SMALL_CODE + for (r = 0; ; r++) { + rk += 4; + t0 = + Td0(byte(s0, 3)) ^ + Td1(byte(s3, 2)) ^ + Td2(byte(s2, 1)) ^ + Td3(byte(s1, 0)) ^ + rk[0]; + t1 = + Td0(byte(s1, 3)) ^ + Td1(byte(s0, 2)) ^ + Td2(byte(s3, 1)) ^ + Td3(byte(s2, 0)) ^ + rk[1]; + t2 = + Td0(byte(s2, 3)) ^ + Td1(byte(s1, 2)) ^ + Td2(byte(s0, 1)) ^ + Td3(byte(s3, 0)) ^ + rk[2]; + t3 = + Td0(byte(s3, 3)) ^ + Td1(byte(s2, 2)) ^ + Td2(byte(s1, 1)) ^ + Td3(byte(s0, 0)) ^ + rk[3]; + if (r == Nr-2) { + break; + } + s0 = t0; s1 = t1; s2 = t2; s3 = t3; + } + rk += 4; + +#else + + /* + * Nr - 1 full rounds: + */ + r = Nr >> 1; + for (;;) { + + t0 = + Td0(byte(s0, 3)) ^ + Td1(byte(s3, 2)) ^ + Td2(byte(s2, 1)) ^ + Td3(byte(s1, 0)) ^ + rk[4]; + t1 = + Td0(byte(s1, 3)) ^ + Td1(byte(s0, 2)) ^ + Td2(byte(s3, 1)) ^ + Td3(byte(s2, 0)) ^ + rk[5]; + t2 = + Td0(byte(s2, 3)) ^ + Td1(byte(s1, 2)) ^ + Td2(byte(s0, 1)) ^ + Td3(byte(s3, 0)) ^ + rk[6]; + t3 = + Td0(byte(s3, 3)) ^ + Td1(byte(s2, 2)) ^ + Td2(byte(s1, 1)) ^ + Td3(byte(s0, 0)) ^ + rk[7]; + + rk += 8; + if (--r == 0) { + break; + } + + + s0 = + Td0(byte(t0, 3)) ^ + Td1(byte(t3, 2)) ^ + Td2(byte(t2, 1)) ^ + Td3(byte(t1, 0)) ^ + rk[0]; + s1 = + Td0(byte(t1, 3)) ^ + Td1(byte(t0, 2)) ^ + Td2(byte(t3, 1)) ^ + Td3(byte(t2, 0)) ^ + rk[1]; + s2 = + Td0(byte(t2, 3)) ^ + Td1(byte(t1, 2)) ^ + Td2(byte(t0, 1)) ^ + Td3(byte(t3, 0)) ^ + rk[2]; + s3 = + Td0(byte(t3, 3)) ^ + Td1(byte(t2, 2)) ^ + Td2(byte(t1, 1)) ^ + Td3(byte(t0, 0)) ^ + rk[3]; + } +#endif + + /* + * apply last round and + * map cipher state to byte array block: + */ + s0 = + (Td4[byte(t0, 3)] & 0xff000000) ^ + (Td4[byte(t3, 2)] & 0x00ff0000) ^ + (Td4[byte(t2, 1)] & 0x0000ff00) ^ + (Td4[byte(t1, 0)] & 0x000000ff) ^ + rk[0]; + STORE32H(s0, pt); + s1 = + (Td4[byte(t1, 3)] & 0xff000000) ^ + (Td4[byte(t0, 2)] & 0x00ff0000) ^ + (Td4[byte(t3, 1)] & 0x0000ff00) ^ + (Td4[byte(t2, 0)] & 0x000000ff) ^ + rk[1]; + STORE32H(s1, pt+4); + s2 = + (Td4[byte(t2, 3)] & 0xff000000) ^ + (Td4[byte(t1, 2)] & 0x00ff0000) ^ + (Td4[byte(t0, 1)] & 0x0000ff00) ^ + (Td4[byte(t3, 0)] & 0x000000ff) ^ + rk[2]; + STORE32H(s2, pt+8); + s3 = + (Td4[byte(t3, 3)] & 0xff000000) ^ + (Td4[byte(t2, 2)] & 0x00ff0000) ^ + (Td4[byte(t1, 1)] & 0x0000ff00) ^ + (Td4[byte(t0, 0)] & 0x000000ff) ^ + rk[3]; + STORE32H(s3, pt+12); + + return CRYPT_OK; +} + + +#ifdef LTC_CLEAN_STACK +int ECB_DEC(const unsigned char *ct, unsigned char *pt, symmetric_key *skey) +{ + int err = _rijndael_ecb_decrypt(ct, pt, skey); + burn_stack(sizeof(unsigned long)*8 + sizeof(unsigned long*) + sizeof(int)*2); + return err; +} +#endif + +/** + Performs a self-test of the AES block cipher + @return CRYPT_OK if functional, CRYPT_NOP if self-test has been disabled +*/ +int ECB_TEST(void) +{ + #ifndef LTC_TEST + return CRYPT_NOP; + #else + int err; + static const struct { + int keylen; + unsigned char key[32], pt[16], ct[16]; + } tests[] = { + { 16, + { 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f }, + { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, + 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff }, + { 0x69, 0xc4, 0xe0, 0xd8, 0x6a, 0x7b, 0x04, 0x30, + 0xd8, 0xcd, 0xb7, 0x80, 0x70, 0xb4, 0xc5, 0x5a } + }, { + 24, + { 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, + 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17 }, + { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, + 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff }, + { 0xdd, 0xa9, 0x7c, 0xa4, 0x86, 0x4c, 0xdf, 0xe0, + 0x6e, 0xaf, 0x70, 0xa0, 0xec, 0x0d, 0x71, 0x91 } + }, { + 32, + { 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, + 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, + 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f }, + { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, + 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff }, + { 0x8e, 0xa2, 0xb7, 0xca, 0x51, 0x67, 0x45, 0xbf, + 0xea, 0xfc, 0x49, 0x90, 0x4b, 0x49, 0x60, 0x89 } + } + }; + + symmetric_key key; + unsigned char tmp[2][16]; + int i, y; + + for (i = 0; i < (int)(sizeof(tests)/sizeof(tests[0])); i++) { + zeromem(&key, sizeof(key)); + if ((err = rijndael_setup(tests[i].key, tests[i].keylen, 0, &key)) != CRYPT_OK) { + return err; + } + + rijndael_ecb_encrypt(tests[i].pt, tmp[0], &key); + rijndael_ecb_decrypt(tmp[0], tmp[1], &key); + if (XMEMCMP(tmp[0], tests[i].ct, 16) || XMEMCMP(tmp[1], tests[i].pt, 16)) { +#if 0 + printf("\n\nTest %d failed\n", i); + if (XMEMCMP(tmp[0], tests[i].ct, 16)) { + printf("CT: "); + for (i = 0; i < 16; i++) { + printf("%02x ", tmp[0][i]); + } + printf("\n"); + } else { + printf("PT: "); + for (i = 0; i < 16; i++) { + printf("%02x ", tmp[1][i]); + } + printf("\n"); + } +#endif + return CRYPT_FAIL_TESTVECTOR; + } + + /* now see if we can encrypt all zero bytes 1000 times, decrypt and come back where we started */ + for (y = 0; y < 16; y++) tmp[0][y] = 0; + for (y = 0; y < 1000; y++) rijndael_ecb_encrypt(tmp[0], tmp[0], &key); + for (y = 0; y < 1000; y++) rijndael_ecb_decrypt(tmp[0], tmp[0], &key); + for (y = 0; y < 16; y++) if (tmp[0][y] != 0) return CRYPT_FAIL_TESTVECTOR; + } + return CRYPT_OK; + #endif +} + +#endif /* ENCRYPT_ONLY */ + + +/** Terminate the context + @param skey The scheduled key +*/ +void ECB_DONE(symmetric_key *skey) +{ + //LTC_UNUSED_PARAM(skey); +} + + +/** + Gets suitable key size + @param keysize [in/out] The length of the recommended key (in bytes). This function will store the suitable size back in this variable. + @return CRYPT_OK if the input key size is acceptable. +*/ +int ECB_KS(int *keysize) +{ + LTC_ARGCHK(keysize != NULL); + + if (*keysize < 16) + return CRYPT_INVALID_KEYSIZE; + if (*keysize < 24) { + *keysize = 16; + return CRYPT_OK; + } else if (*keysize < 32) { + *keysize = 24; + return CRYPT_OK; + } else { + *keysize = 32; + return CRYPT_OK; + } +} + +#endif + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file cbc_decrypt.c + CBC implementation, encrypt block, Tom St Denis +*/ + + +#ifdef LTC_CBC_MODE + +/** + CBC decrypt + @param ct Ciphertext + @param pt [out] Plaintext + @param len The number of bytes to process (must be multiple of block length) + @param cbc CBC state + @return CRYPT_OK if successful +*/ +int cbc_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_CBC *cbc) +{ + int x, err; + unsigned char tmp[16]; +#ifdef LTC_FAST + LTC_FAST_TYPE tmpy; +#else + unsigned char tmpy; +#endif + + LTC_ARGCHK(pt != NULL); + LTC_ARGCHK(ct != NULL); + LTC_ARGCHK(cbc != NULL); + + if ((err = cipher_is_valid(cbc->cipher)) != CRYPT_OK) { + return err; + } + + /* is blocklen valid? */ + if (cbc->blocklen < 1 || cbc->blocklen > (int)sizeof(cbc->IV)) { + return CRYPT_INVALID_ARG; + } + + if (len % cbc->blocklen) { + return CRYPT_INVALID_ARG; + } +#ifdef LTC_FAST + if (cbc->blocklen % sizeof(LTC_FAST_TYPE)) { + return CRYPT_INVALID_ARG; + } +#endif + + if (cipher_descriptor[cbc->cipher].accel_cbc_decrypt != NULL) { + return cipher_descriptor[cbc->cipher].accel_cbc_decrypt(ct, pt, len / cbc->blocklen, cbc->IV, &cbc->key); + } else { + while (len) { + /* decrypt */ + if ((err = cipher_descriptor[cbc->cipher].ecb_decrypt(ct, tmp, &cbc->key)) != CRYPT_OK) { + return err; + } + + /* xor IV against plaintext */ + #if defined(LTC_FAST) + for (x = 0; x < cbc->blocklen; x += sizeof(LTC_FAST_TYPE)) { + tmpy = *((LTC_FAST_TYPE*)((unsigned char *)cbc->IV + x)) ^ *((LTC_FAST_TYPE*)((unsigned char *)tmp + x)); + *((LTC_FAST_TYPE*)((unsigned char *)cbc->IV + x)) = *((LTC_FAST_TYPE*)((unsigned char *)ct + x)); + *((LTC_FAST_TYPE*)((unsigned char *)pt + x)) = tmpy; + } + #else + for (x = 0; x < cbc->blocklen; x++) { + tmpy = tmp[x] ^ cbc->IV[x]; + cbc->IV[x] = ct[x]; + pt[x] = tmpy; + } + #endif + + ct += cbc->blocklen; + pt += cbc->blocklen; + len -= cbc->blocklen; + } + } + return CRYPT_OK; +} + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file cbc_done.c + CBC implementation, finish chain, Tom St Denis +*/ + +#ifdef LTC_CBC_MODE + +/** Terminate the chain + @param cbc The CBC chain to terminate + @return CRYPT_OK on success +*/ +int cbc_done(symmetric_CBC *cbc) +{ + int err; + LTC_ARGCHK(cbc != NULL); + + if ((err = cipher_is_valid(cbc->cipher)) != CRYPT_OK) { + return err; + } + cipher_descriptor[cbc->cipher].done(&cbc->key); + return CRYPT_OK; +} + + + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file cbc_encrypt.c + CBC implementation, encrypt block, Tom St Denis +*/ + + +#ifdef LTC_CBC_MODE + +/** + CBC encrypt + @param pt Plaintext + @param ct [out] Ciphertext + @param len The number of bytes to process (must be multiple of block length) + @param cbc CBC state + @return CRYPT_OK if successful +*/ +int cbc_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_CBC *cbc) +{ + int x, err; + + LTC_ARGCHK(pt != NULL); + LTC_ARGCHK(ct != NULL); + LTC_ARGCHK(cbc != NULL); + + if ((err = cipher_is_valid(cbc->cipher)) != CRYPT_OK) { + return err; + } + + /* is blocklen valid? */ + if (cbc->blocklen < 1 || cbc->blocklen > (int)sizeof(cbc->IV)) { + return CRYPT_INVALID_ARG; + } + + if (len % cbc->blocklen) { + return CRYPT_INVALID_ARG; + } +#ifdef LTC_FAST + if (cbc->blocklen % sizeof(LTC_FAST_TYPE)) { + return CRYPT_INVALID_ARG; + } +#endif + + if (cipher_descriptor[cbc->cipher].accel_cbc_encrypt != NULL) { + return cipher_descriptor[cbc->cipher].accel_cbc_encrypt(pt, ct, len / cbc->blocklen, cbc->IV, &cbc->key); + } else { + while (len) { + /* xor IV against plaintext */ + #if defined(LTC_FAST) + for (x = 0; x < cbc->blocklen; x += sizeof(LTC_FAST_TYPE)) { + *((LTC_FAST_TYPE*)((unsigned char *)cbc->IV + x)) ^= *((LTC_FAST_TYPE*)((unsigned char *)pt + x)); + } + #else + for (x = 0; x < cbc->blocklen; x++) { + cbc->IV[x] ^= pt[x]; + } + #endif + + /* encrypt */ + if ((err = cipher_descriptor[cbc->cipher].ecb_encrypt(cbc->IV, ct, &cbc->key)) != CRYPT_OK) { + return err; + } + + /* store IV [ciphertext] for a future block */ + #if defined(LTC_FAST) + for (x = 0; x < cbc->blocklen; x += sizeof(LTC_FAST_TYPE)) { + *((LTC_FAST_TYPE*)((unsigned char *)cbc->IV + x)) = *((LTC_FAST_TYPE*)((unsigned char *)ct + x)); + } + #else + for (x = 0; x < cbc->blocklen; x++) { + cbc->IV[x] = ct[x]; + } + #endif + + ct += cbc->blocklen; + pt += cbc->blocklen; + len -= cbc->blocklen; + } + } + return CRYPT_OK; +} + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file cbc_getiv.c + CBC implementation, get IV, Tom St Denis +*/ + +#ifdef LTC_CBC_MODE + +/** + Get the current initial vector + @param IV [out] The destination of the initial vector + @param len [in/out] The max size and resulting size of the initial vector + @param cbc The CBC state + @return CRYPT_OK if successful +*/ +int cbc_getiv(unsigned char *IV, unsigned long *len, symmetric_CBC *cbc) +{ + LTC_ARGCHK(IV != NULL); + LTC_ARGCHK(len != NULL); + LTC_ARGCHK(cbc != NULL); + if ((unsigned long)cbc->blocklen > *len) { + *len = cbc->blocklen; + return CRYPT_BUFFER_OVERFLOW; + } + XMEMCPY(IV, cbc->IV, cbc->blocklen); + *len = cbc->blocklen; + + return CRYPT_OK; +} + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file cbc_setiv.c + CBC implementation, set IV, Tom St Denis +*/ + + +#ifdef LTC_CBC_MODE + +/** + Set an initial vector + @param IV The initial vector + @param len The length of the vector (in octets) + @param cbc The CBC state + @return CRYPT_OK if successful +*/ +int cbc_setiv(const unsigned char *IV, unsigned long len, symmetric_CBC *cbc) +{ + LTC_ARGCHK(IV != NULL); + LTC_ARGCHK(cbc != NULL); + if (len != (unsigned long)cbc->blocklen) { + return CRYPT_INVALID_ARG; + } + XMEMCPY(cbc->IV, IV, len); + return CRYPT_OK; +} + +#endif + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + +/** + @file cbc_start.c + CBC implementation, start chain, Tom St Denis +*/ + +#ifdef LTC_CBC_MODE + +/** + Initialize a CBC context + @param cipher The index of the cipher desired + @param IV The initial vector + @param key The secret key + @param keylen The length of the secret key (octets) + @param num_rounds Number of rounds in the cipher desired (0 for default) + @param cbc The CBC state to initialize + @return CRYPT_OK if successful +*/ +int cbc_start(int cipher, const unsigned char *IV, const unsigned char *key, + int keylen, int num_rounds, symmetric_CBC *cbc) +{ + int x, err; + + LTC_ARGCHK(IV != NULL); + LTC_ARGCHK(key != NULL); + LTC_ARGCHK(cbc != NULL); + + /* bad param? */ + if ((err = cipher_is_valid(cipher)) != CRYPT_OK) { + return err; + } + + /* setup cipher */ + if ((err = cipher_descriptor[cipher].setup(key, keylen, num_rounds, &cbc->key)) != CRYPT_OK) { + return err; + } + + /* copy IV */ + cbc->blocklen = cipher_descriptor[cipher].block_length; + cbc->cipher = cipher; + for (x = 0; x < cbc->blocklen; x++) { + cbc->IV[x] = IV[x]; + } + return CRYPT_OK; +} + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file gcm_add_iv.c + GCM implementation, add IV data to the state, by Tom St Denis +*/ + + +#ifdef LTC_GCM_MODE + +/** + Add IV data to the GCM state + @param gcm The GCM state + @param IV The initial value data to add + @param IVlen The length of the IV + @return CRYPT_OK on success + */ +int gcm_add_iv(gcm_state *gcm, + const unsigned char *IV, unsigned long IVlen) +{ + unsigned long x, y; + int err; + + LTC_ARGCHK(gcm != NULL); + if (IVlen > 0) { + LTC_ARGCHK(IV != NULL); + } + + /* must be in IV mode */ + if (gcm->mode != LTC_GCM_MODE_IV) { + return CRYPT_INVALID_ARG; + } + + if (gcm->buflen >= 16 || gcm->buflen < 0) { + return CRYPT_INVALID_ARG; + } + + if ((err = cipher_is_valid(gcm->cipher)) != CRYPT_OK) { + return err; + } + + + /* trip the ivmode flag */ + if (IVlen + gcm->buflen > 12) { + gcm->ivmode |= 1; + } + + x = 0; +#ifdef LTC_FAST + if (gcm->buflen == 0) { + for (x = 0; x < (IVlen & ~15); x += 16) { + for (y = 0; y < 16; y += sizeof(LTC_FAST_TYPE)) { + *((LTC_FAST_TYPE*)(&gcm->X[y])) ^= *((LTC_FAST_TYPE*)(&IV[x + y])); + } + gcm_mult_h(gcm, gcm->X); + gcm->totlen += 128; + } + IV += x; + } +#endif + + /* start adding IV data to the state */ + for (; x < IVlen; x++) { + gcm->buf[gcm->buflen++] = *IV++; + + if (gcm->buflen == 16) { + /* GF mult it */ + for (y = 0; y < 16; y++) { + gcm->X[y] ^= gcm->buf[y]; + } + gcm_mult_h(gcm, gcm->X); + gcm->buflen = 0; + gcm->totlen += 128; + } + } + + return CRYPT_OK; +} + +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/encauth/gcm/gcm_add_iv.c,v $ */ +/* $Revision: 1.9 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file gcm_done.c + GCM implementation, Terminate the stream, by Tom St Denis +*/ + + +#ifdef LTC_GCM_MODE + +/** + Terminate a GCM stream + @param gcm The GCM state + @param tag [out] The destination for the MAC tag + @param taglen [in/out] The length of the MAC tag + @return CRYPT_OK on success + */ +int gcm_done(gcm_state *gcm, + unsigned char *tag, unsigned long *taglen) +{ + unsigned long x; + int err; + + LTC_ARGCHK(gcm != NULL); + LTC_ARGCHK(tag != NULL); + LTC_ARGCHK(taglen != NULL); + + if (gcm->buflen > 16 || gcm->buflen < 0) { + return CRYPT_INVALID_ARG; + } + + if ((err = cipher_is_valid(gcm->cipher)) != CRYPT_OK) { + return err; + } + + + if (gcm->mode != LTC_GCM_MODE_TEXT) { + return CRYPT_INVALID_ARG; + } + + /* handle remaining ciphertext */ + if (gcm->buflen) { + gcm->pttotlen += gcm->buflen * CONST64(8); + gcm_mult_h(gcm, gcm->X); + } + + /* length */ + STORE64H(gcm->totlen, gcm->buf); + STORE64H(gcm->pttotlen, gcm->buf+8); + for (x = 0; x < 16; x++) { + gcm->X[x] ^= gcm->buf[x]; + } + gcm_mult_h(gcm, gcm->X); + + /* encrypt original counter */ + if ((err = cipher_descriptor[gcm->cipher].ecb_encrypt(gcm->Y_0, gcm->buf, &gcm->K)) != CRYPT_OK) { + return err; + } + for (x = 0; x < 16 && x < *taglen; x++) { + tag[x] = gcm->buf[x] ^ gcm->X[x]; + } + *taglen = x; + + cipher_descriptor[gcm->cipher].done(&gcm->K); + + return CRYPT_OK; +} + +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/encauth/gcm/gcm_done.c,v $ */ +/* $Revision: 1.11 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file gcm_init.c + GCM implementation, initialize state, by Tom St Denis +*/ + + +#ifdef LTC_GCM_MODE + +/** + Initialize a GCM state + @param gcm The GCM state to initialize + @param cipher The index of the cipher to use + @param key The secret key + @param keylen The length of the secret key + @return CRYPT_OK on success + */ +int gcm_init(gcm_state *gcm, int cipher, + const unsigned char *key, int keylen) +{ + int err; + unsigned char B[16]; +#ifdef LTC_GCM_TABLES + int x, y, z, t; +#endif + + LTC_ARGCHK(gcm != NULL); + LTC_ARGCHK(key != NULL); + +#ifdef LTC_FAST + if (16 % sizeof(LTC_FAST_TYPE)) { + return CRYPT_INVALID_ARG; + } +#endif + + /* is cipher valid? */ + if ((err = cipher_is_valid(cipher)) != CRYPT_OK) { + return err; + } + if (cipher_descriptor[cipher].block_length != 16) { + return CRYPT_INVALID_CIPHER; + } + + /* schedule key */ + if ((err = cipher_descriptor[cipher].setup(key, keylen, 0, &gcm->K)) != CRYPT_OK) { + return err; + } + + /* H = E(0) */ + zeromem(B, 16); + if ((err = cipher_descriptor[cipher].ecb_encrypt(B, gcm->H, &gcm->K)) != CRYPT_OK) { + return err; + } + + /* setup state */ + zeromem(gcm->buf, sizeof(gcm->buf)); + zeromem(gcm->X, sizeof(gcm->X)); + gcm->cipher = cipher; + gcm->mode = LTC_GCM_MODE_IV; + gcm->ivmode = 0; + gcm->buflen = 0; + gcm->totlen = 0; + gcm->pttotlen = 0; + +#ifdef LTC_GCM_TABLES + /* setup tables */ + + /* generate the first table as it has no shifting (from which we make the other tables) */ + zeromem(B, 16); + for (y = 0; y < 256; y++) { + B[0] = y; + gcm_gf_mult(gcm->H, B, &gcm->PC[0][y][0]); + } + + /* now generate the rest of the tables based the previous table */ + for (x = 1; x < 16; x++) { + for (y = 0; y < 256; y++) { + /* now shift it right by 8 bits */ + t = gcm->PC[x-1][y][15]; + for (z = 15; z > 0; z--) { + gcm->PC[x][y][z] = gcm->PC[x-1][y][z-1]; + } + gcm->PC[x][y][0] = gcm_shift_table[t<<1]; + gcm->PC[x][y][1] ^= gcm_shift_table[(t<<1)+1]; + } + } + +#endif + + return CRYPT_OK; +} + +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/encauth/gcm/gcm_init.c,v $ */ +/* $Revision: 1.20 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file gcm_process.c + GCM implementation, process message data, by Tom St Denis +*/ + + +#ifdef LTC_GCM_MODE + +/** + Process plaintext/ciphertext through GCM + @param gcm The GCM state + @param pt The plaintext + @param ptlen The plaintext length (ciphertext length is the same) + @param ct The ciphertext + @param direction Encrypt or Decrypt mode (GCM_ENCRYPT or GCM_DECRYPT) + @return CRYPT_OK on success + */ +int gcm_process(gcm_state *gcm, + unsigned char *pt, unsigned long ptlen, + unsigned char *ct, + int direction) +{ + unsigned long x; + int y, err; + unsigned char b; + + LTC_ARGCHK(gcm != NULL); + if (ptlen > 0) { + LTC_ARGCHK(pt != NULL); + LTC_ARGCHK(ct != NULL); + } + + if (gcm->buflen > 16 || gcm->buflen < 0) { + return CRYPT_INVALID_ARG; + } + + if ((err = cipher_is_valid(gcm->cipher)) != CRYPT_OK) { + return err; + } + + /* in AAD mode? */ + if (gcm->mode == LTC_GCM_MODE_AAD) { + /* let's process the AAD */ + if (gcm->buflen) { + gcm->totlen += gcm->buflen * CONST64(8); + gcm_mult_h(gcm, gcm->X); + } + + /* increment counter */ + for (y = 15; y >= 12; y--) { + if (++gcm->Y[y] & 255) { break; } + } + /* encrypt the counter */ + if ((err = cipher_descriptor[gcm->cipher].ecb_encrypt(gcm->Y, gcm->buf, &gcm->K)) != CRYPT_OK) { + return err; + } + + gcm->buflen = 0; + gcm->mode = LTC_GCM_MODE_TEXT; + } + + if (gcm->mode != LTC_GCM_MODE_TEXT) { + return CRYPT_INVALID_ARG; + } + + x = 0; +#ifdef LTC_FAST + if (gcm->buflen == 0) { + if (direction == GCM_ENCRYPT) { + for (x = 0; x < (ptlen & ~15); x += 16) { + /* ctr encrypt */ + for (y = 0; y < 16; y += sizeof(LTC_FAST_TYPE)) { + *((LTC_FAST_TYPE*)(&ct[x + y])) = *((LTC_FAST_TYPE*)(&pt[x+y])) ^ *((LTC_FAST_TYPE*)(&gcm->buf[y])); + *((LTC_FAST_TYPE*)(&gcm->X[y])) ^= *((LTC_FAST_TYPE*)(&ct[x+y])); + } + /* GMAC it */ + gcm->pttotlen += 128; + gcm_mult_h(gcm, gcm->X); + /* increment counter */ + for (y = 15; y >= 12; y--) { + if (++gcm->Y[y] & 255) { break; } + } + if ((err = cipher_descriptor[gcm->cipher].ecb_encrypt(gcm->Y, gcm->buf, &gcm->K)) != CRYPT_OK) { + return err; + } + } + } else { + for (x = 0; x < (ptlen & ~15); x += 16) { + /* ctr encrypt */ + for (y = 0; y < 16; y += sizeof(LTC_FAST_TYPE)) { + *((LTC_FAST_TYPE*)(&gcm->X[y])) ^= *((LTC_FAST_TYPE*)(&ct[x+y])); + *((LTC_FAST_TYPE*)(&pt[x + y])) = *((LTC_FAST_TYPE*)(&ct[x+y])) ^ *((LTC_FAST_TYPE*)(&gcm->buf[y])); + } + /* GMAC it */ + gcm->pttotlen += 128; + gcm_mult_h(gcm, gcm->X); + /* increment counter */ + for (y = 15; y >= 12; y--) { + if (++gcm->Y[y] & 255) { break; } + } + if ((err = cipher_descriptor[gcm->cipher].ecb_encrypt(gcm->Y, gcm->buf, &gcm->K)) != CRYPT_OK) { + return err; + } + } + } + } +#endif + + /* process text */ + for (; x < ptlen; x++) { + if (gcm->buflen == 16) { + gcm->pttotlen += 128; + gcm_mult_h(gcm, gcm->X); + + /* increment counter */ + for (y = 15; y >= 12; y--) { + if (++gcm->Y[y] & 255) { break; } + } + if ((err = cipher_descriptor[gcm->cipher].ecb_encrypt(gcm->Y, gcm->buf, &gcm->K)) != CRYPT_OK) { + return err; + } + gcm->buflen = 0; + } + + if (direction == GCM_ENCRYPT) { + b = ct[x] = pt[x] ^ gcm->buf[gcm->buflen]; + } else { + b = ct[x]; + pt[x] = ct[x] ^ gcm->buf[gcm->buflen]; + } + gcm->X[gcm->buflen++] ^= b; + } + + return CRYPT_OK; +} + +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/encauth/gcm/gcm_process.c,v $ */ +/* $Revision: 1.16 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file gcm_mult_h.c + GCM implementation, do the GF mult, by Tom St Denis +*/ + + +#if defined(LTC_GCM_MODE) +/** + GCM multiply by H + @param gcm The GCM state which holds the H value + @param I The value to multiply H by + */ +void gcm_mult_h(gcm_state *gcm, unsigned char *I) +{ + unsigned char T[16]; +#ifdef LTC_GCM_TABLES + int x, y; +#ifdef LTC_GCM_TABLES_SSE2 + asm("movdqa (%0),%%xmm0"::"r"(&gcm->PC[0][I[0]][0])); + for (x = 1; x < 16; x++) { + asm("pxor (%0),%%xmm0"::"r"(&gcm->PC[x][I[x]][0])); + } + asm("movdqa %%xmm0,(%0)"::"r"(&T)); +#else + XMEMCPY(T, &gcm->PC[0][I[0]][0], 16); + for (x = 1; x < 16; x++) { +#ifdef LTC_FAST + for (y = 0; y < 16; y += sizeof(LTC_FAST_TYPE)) { + *((LTC_FAST_TYPE *)(T + y)) ^= *((LTC_FAST_TYPE *)(&gcm->PC[x][I[x]][y])); + } +#else + for (y = 0; y < 16; y++) { + T[y] ^= gcm->PC[x][I[x]][y]; + } +#endif /* LTC_FAST */ + } +#endif /* LTC_GCM_TABLES_SSE2 */ +#else + gcm_gf_mult(gcm->H, I, T); +#endif + XMEMCPY(I, T, 16); +} +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/encauth/gcm/gcm_mult_h.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file gcm_gf_mult.c + GCM implementation, do the GF mult, by Tom St Denis +*/ + + +#if defined(LTC_GCM_TABLES) || defined(LRW_TABLES) || ((defined(LTC_GCM_MODE) || defined(LTC_GCM_MODE)) && defined(LTC_FAST)) + +/* this is x*2^128 mod p(x) ... the results are 16 bytes each stored in a packed format. Since only the + * lower 16 bits are not zero'ed I removed the upper 14 bytes */ +const unsigned char gcm_shift_table[256*2] = { +0x00, 0x00, 0x01, 0xc2, 0x03, 0x84, 0x02, 0x46, 0x07, 0x08, 0x06, 0xca, 0x04, 0x8c, 0x05, 0x4e, +0x0e, 0x10, 0x0f, 0xd2, 0x0d, 0x94, 0x0c, 0x56, 0x09, 0x18, 0x08, 0xda, 0x0a, 0x9c, 0x0b, 0x5e, +0x1c, 0x20, 0x1d, 0xe2, 0x1f, 0xa4, 0x1e, 0x66, 0x1b, 0x28, 0x1a, 0xea, 0x18, 0xac, 0x19, 0x6e, +0x12, 0x30, 0x13, 0xf2, 0x11, 0xb4, 0x10, 0x76, 0x15, 0x38, 0x14, 0xfa, 0x16, 0xbc, 0x17, 0x7e, +0x38, 0x40, 0x39, 0x82, 0x3b, 0xc4, 0x3a, 0x06, 0x3f, 0x48, 0x3e, 0x8a, 0x3c, 0xcc, 0x3d, 0x0e, +0x36, 0x50, 0x37, 0x92, 0x35, 0xd4, 0x34, 0x16, 0x31, 0x58, 0x30, 0x9a, 0x32, 0xdc, 0x33, 0x1e, +0x24, 0x60, 0x25, 0xa2, 0x27, 0xe4, 0x26, 0x26, 0x23, 0x68, 0x22, 0xaa, 0x20, 0xec, 0x21, 0x2e, +0x2a, 0x70, 0x2b, 0xb2, 0x29, 0xf4, 0x28, 0x36, 0x2d, 0x78, 0x2c, 0xba, 0x2e, 0xfc, 0x2f, 0x3e, +0x70, 0x80, 0x71, 0x42, 0x73, 0x04, 0x72, 0xc6, 0x77, 0x88, 0x76, 0x4a, 0x74, 0x0c, 0x75, 0xce, +0x7e, 0x90, 0x7f, 0x52, 0x7d, 0x14, 0x7c, 0xd6, 0x79, 0x98, 0x78, 0x5a, 0x7a, 0x1c, 0x7b, 0xde, +0x6c, 0xa0, 0x6d, 0x62, 0x6f, 0x24, 0x6e, 0xe6, 0x6b, 0xa8, 0x6a, 0x6a, 0x68, 0x2c, 0x69, 0xee, +0x62, 0xb0, 0x63, 0x72, 0x61, 0x34, 0x60, 0xf6, 0x65, 0xb8, 0x64, 0x7a, 0x66, 0x3c, 0x67, 0xfe, +0x48, 0xc0, 0x49, 0x02, 0x4b, 0x44, 0x4a, 0x86, 0x4f, 0xc8, 0x4e, 0x0a, 0x4c, 0x4c, 0x4d, 0x8e, +0x46, 0xd0, 0x47, 0x12, 0x45, 0x54, 0x44, 0x96, 0x41, 0xd8, 0x40, 0x1a, 0x42, 0x5c, 0x43, 0x9e, +0x54, 0xe0, 0x55, 0x22, 0x57, 0x64, 0x56, 0xa6, 0x53, 0xe8, 0x52, 0x2a, 0x50, 0x6c, 0x51, 0xae, +0x5a, 0xf0, 0x5b, 0x32, 0x59, 0x74, 0x58, 0xb6, 0x5d, 0xf8, 0x5c, 0x3a, 0x5e, 0x7c, 0x5f, 0xbe, +0xe1, 0x00, 0xe0, 0xc2, 0xe2, 0x84, 0xe3, 0x46, 0xe6, 0x08, 0xe7, 0xca, 0xe5, 0x8c, 0xe4, 0x4e, +0xef, 0x10, 0xee, 0xd2, 0xec, 0x94, 0xed, 0x56, 0xe8, 0x18, 0xe9, 0xda, 0xeb, 0x9c, 0xea, 0x5e, +0xfd, 0x20, 0xfc, 0xe2, 0xfe, 0xa4, 0xff, 0x66, 0xfa, 0x28, 0xfb, 0xea, 0xf9, 0xac, 0xf8, 0x6e, +0xf3, 0x30, 0xf2, 0xf2, 0xf0, 0xb4, 0xf1, 0x76, 0xf4, 0x38, 0xf5, 0xfa, 0xf7, 0xbc, 0xf6, 0x7e, +0xd9, 0x40, 0xd8, 0x82, 0xda, 0xc4, 0xdb, 0x06, 0xde, 0x48, 0xdf, 0x8a, 0xdd, 0xcc, 0xdc, 0x0e, +0xd7, 0x50, 0xd6, 0x92, 0xd4, 0xd4, 0xd5, 0x16, 0xd0, 0x58, 0xd1, 0x9a, 0xd3, 0xdc, 0xd2, 0x1e, +0xc5, 0x60, 0xc4, 0xa2, 0xc6, 0xe4, 0xc7, 0x26, 0xc2, 0x68, 0xc3, 0xaa, 0xc1, 0xec, 0xc0, 0x2e, +0xcb, 0x70, 0xca, 0xb2, 0xc8, 0xf4, 0xc9, 0x36, 0xcc, 0x78, 0xcd, 0xba, 0xcf, 0xfc, 0xce, 0x3e, +0x91, 0x80, 0x90, 0x42, 0x92, 0x04, 0x93, 0xc6, 0x96, 0x88, 0x97, 0x4a, 0x95, 0x0c, 0x94, 0xce, +0x9f, 0x90, 0x9e, 0x52, 0x9c, 0x14, 0x9d, 0xd6, 0x98, 0x98, 0x99, 0x5a, 0x9b, 0x1c, 0x9a, 0xde, +0x8d, 0xa0, 0x8c, 0x62, 0x8e, 0x24, 0x8f, 0xe6, 0x8a, 0xa8, 0x8b, 0x6a, 0x89, 0x2c, 0x88, 0xee, +0x83, 0xb0, 0x82, 0x72, 0x80, 0x34, 0x81, 0xf6, 0x84, 0xb8, 0x85, 0x7a, 0x87, 0x3c, 0x86, 0xfe, +0xa9, 0xc0, 0xa8, 0x02, 0xaa, 0x44, 0xab, 0x86, 0xae, 0xc8, 0xaf, 0x0a, 0xad, 0x4c, 0xac, 0x8e, +0xa7, 0xd0, 0xa6, 0x12, 0xa4, 0x54, 0xa5, 0x96, 0xa0, 0xd8, 0xa1, 0x1a, 0xa3, 0x5c, 0xa2, 0x9e, +0xb5, 0xe0, 0xb4, 0x22, 0xb6, 0x64, 0xb7, 0xa6, 0xb2, 0xe8, 0xb3, 0x2a, 0xb1, 0x6c, 0xb0, 0xae, +0xbb, 0xf0, 0xba, 0x32, 0xb8, 0x74, 0xb9, 0xb6, 0xbc, 0xf8, 0xbd, 0x3a, 0xbf, 0x7c, 0xbe, 0xbe }; + +#endif + + +#if defined(LTC_GCM_MODE) || defined(LRW_MODE) + +#ifndef LTC_FAST +/* right shift */ +static void gcm_rightshift(unsigned char *a) +{ + int x; + for (x = 15; x > 0; x--) { + a[x] = (a[x]>>1) | ((a[x-1]<<7)&0x80); + } + a[0] >>= 1; +} + +/* c = b*a */ +static const unsigned char mask[] = { 0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01 }; +static const unsigned char poly[] = { 0x00, 0xE1 }; + + +/** + GCM GF multiplier (internal use only) bitserial + @param a First value + @param b Second value + @param c Destination for a * b + */ +void gcm_gf_mult(const unsigned char *a, const unsigned char *b, unsigned char *c) +{ + unsigned char Z[16], V[16]; + unsigned char x, y, z; + + zeromem(Z, 16); + XMEMCPY(V, a, 16); + for (x = 0; x < 128; x++) { + if (b[x>>3] & mask[x&7]) { + for (y = 0; y < 16; y++) { + Z[y] ^= V[y]; + } + } + z = V[15] & 0x01; + gcm_rightshift(V); + V[0] ^= poly[z]; + } + XMEMCPY(c, Z, 16); +} + +#else + +/* map normal numbers to "ieee" way ... e.g. bit reversed */ +#define M(x) ( ((x&8)>>3) | ((x&4)>>1) | ((x&2)<<1) | ((x&1)<<3) ) + +#define BPD (sizeof(LTC_FAST_TYPE) * 8) +#define WPV (1 + (16 / sizeof(LTC_FAST_TYPE))) + +/** + GCM GF multiplier (internal use only) word oriented + @param a First value + @param b Second value + @param c Destination for a * b + */ +void gcm_gf_mult(const unsigned char *a, const unsigned char *b, unsigned char *c) +{ + int i, j, k, u; + LTC_FAST_TYPE B[16][WPV], tmp[32 / sizeof(LTC_FAST_TYPE)], pB[16 / sizeof(LTC_FAST_TYPE)], zz, z; + unsigned char pTmp[32]; + + /* create simple tables */ + zeromem(B[0], sizeof(B[0])); + zeromem(B[M(1)], sizeof(B[M(1)])); + +#ifdef ENDIAN_32BITWORD + for (i = 0; i < 4; i++) { + LOAD32H(B[M(1)][i], a + (i<<2)); + LOAD32L(pB[i], b + (i<<2)); + } +#else + for (i = 0; i < 2; i++) { + LOAD64H(B[M(1)][i], a + (i<<3)); + LOAD64L(pB[i], b + (i<<3)); + } +#endif + + /* now create 2, 4 and 8 */ + B[M(2)][0] = B[M(1)][0] >> 1; + B[M(4)][0] = B[M(1)][0] >> 2; + B[M(8)][0] = B[M(1)][0] >> 3; + for (i = 1; i < (int)WPV; i++) { + B[M(2)][i] = (B[M(1)][i-1] << (BPD-1)) | (B[M(1)][i] >> 1); + B[M(4)][i] = (B[M(1)][i-1] << (BPD-2)) | (B[M(1)][i] >> 2); + B[M(8)][i] = (B[M(1)][i-1] << (BPD-3)) | (B[M(1)][i] >> 3); + } + + /* now all values with two bits which are 3, 5, 6, 9, 10, 12 */ + for (i = 0; i < (int)WPV; i++) { + B[M(3)][i] = B[M(1)][i] ^ B[M(2)][i]; + B[M(5)][i] = B[M(1)][i] ^ B[M(4)][i]; + B[M(6)][i] = B[M(2)][i] ^ B[M(4)][i]; + B[M(9)][i] = B[M(1)][i] ^ B[M(8)][i]; + B[M(10)][i] = B[M(2)][i] ^ B[M(8)][i]; + B[M(12)][i] = B[M(8)][i] ^ B[M(4)][i]; + + /* now all 3 bit values and the only 4 bit value: 7, 11, 13, 14, 15 */ + B[M(7)][i] = B[M(3)][i] ^ B[M(4)][i]; + B[M(11)][i] = B[M(3)][i] ^ B[M(8)][i]; + B[M(13)][i] = B[M(1)][i] ^ B[M(12)][i]; + B[M(14)][i] = B[M(6)][i] ^ B[M(8)][i]; + B[M(15)][i] = B[M(7)][i] ^ B[M(8)][i]; + } + + zeromem(tmp, sizeof(tmp)); + + /* compute product four bits of each word at a time */ + /* for each nibble */ + for (i = (BPD/4)-1; i >= 0; i--) { + /* for each word */ + for (j = 0; j < (int)(WPV-1); j++) { + /* grab the 4 bits recall the nibbles are backwards so it's a shift by (i^1)*4 */ + u = (pB[j] >> ((i^1)<<2)) & 15; + + /* add offset by the word count the table looked up value to the result */ + for (k = 0; k < (int)WPV; k++) { + tmp[k+j] ^= B[u][k]; + } + } + /* shift result up by 4 bits */ + if (i != 0) { + for (z = j = 0; j < (int)(32 / sizeof(LTC_FAST_TYPE)); j++) { + zz = tmp[j] << (BPD-4); + tmp[j] = (tmp[j] >> 4) | z; + z = zz; + } + } + } + + /* store product */ +#ifdef ENDIAN_32BITWORD + for (i = 0; i < 8; i++) { + STORE32H(tmp[i], pTmp + (i<<2)); + } +#else + for (i = 0; i < 4; i++) { + STORE64H(tmp[i], pTmp + (i<<3)); + } +#endif + + /* reduce by taking most significant byte and adding the appropriate two byte sequence 16 bytes down */ + for (i = 31; i >= 16; i--) { + pTmp[i-16] ^= gcm_shift_table[((unsigned)pTmp[i]<<1)]; + pTmp[i-15] ^= gcm_shift_table[((unsigned)pTmp[i]<<1)+1]; + } + + for (i = 0; i < 16; i++) { + c[i] = pTmp[i]; + } + +} + +#endif + +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/encauth/gcm/gcm_gf_mult.c,v $ */ +/* $Revision: 1.25 $ */ +/* $Date: 2007/05/12 14:32:35 $ */ + + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file gcm_add_aad.c + GCM implementation, Add AAD data to the stream, by Tom St Denis +*/ + + +#ifdef LTC_GCM_MODE + +/** + Add AAD to the GCM state + @param gcm The GCM state + @param adata The additional authentication data to add to the GCM state + @param adatalen The length of the AAD data. + @return CRYPT_OK on success + */ +int gcm_add_aad(gcm_state *gcm, + const unsigned char *adata, unsigned long adatalen) +{ + unsigned long x; + int err; +#ifdef LTC_FAST + unsigned long y; +#endif + + LTC_ARGCHK(gcm != NULL); + if (adatalen > 0) { + LTC_ARGCHK(adata != NULL); + } + + if (gcm->buflen > 16 || gcm->buflen < 0) { + return CRYPT_INVALID_ARG; + } + + if ((err = cipher_is_valid(gcm->cipher)) != CRYPT_OK) { + return err; + } + + /* in IV mode? */ + if (gcm->mode == LTC_GCM_MODE_IV) { + /* let's process the IV */ + if (gcm->ivmode || gcm->buflen != 12) { + for (x = 0; x < (unsigned long)gcm->buflen; x++) { + gcm->X[x] ^= gcm->buf[x]; + } + if (gcm->buflen) { + gcm->totlen += gcm->buflen * CONST64(8); + gcm_mult_h(gcm, gcm->X); + } + + /* mix in the length */ + zeromem(gcm->buf, 8); + STORE64H(gcm->totlen, gcm->buf+8); + for (x = 0; x < 16; x++) { + gcm->X[x] ^= gcm->buf[x]; + } + gcm_mult_h(gcm, gcm->X); + + /* copy counter out */ + XMEMCPY(gcm->Y, gcm->X, 16); + zeromem(gcm->X, 16); + } else { + XMEMCPY(gcm->Y, gcm->buf, 12); + gcm->Y[12] = 0; + gcm->Y[13] = 0; + gcm->Y[14] = 0; + gcm->Y[15] = 1; + } + XMEMCPY(gcm->Y_0, gcm->Y, 16); + zeromem(gcm->buf, 16); + gcm->buflen = 0; + gcm->totlen = 0; + gcm->mode = LTC_GCM_MODE_AAD; + } + + if (gcm->mode != LTC_GCM_MODE_AAD || gcm->buflen >= 16) { + return CRYPT_INVALID_ARG; + } + + x = 0; +#ifdef LTC_FAST + if (gcm->buflen == 0) { + for (x = 0; x < (adatalen & ~15); x += 16) { + for (y = 0; y < 16; y += sizeof(LTC_FAST_TYPE)) { + *((LTC_FAST_TYPE*)(&gcm->X[y])) ^= *((LTC_FAST_TYPE*)(&adata[x + y])); + } + gcm_mult_h(gcm, gcm->X); + gcm->totlen += 128; + } + adata += x; + } +#endif + + + /* start adding AAD data to the state */ + for (; x < adatalen; x++) { + gcm->X[gcm->buflen++] ^= *adata++; + + if (gcm->buflen == 16) { + /* GF mult it */ + gcm_mult_h(gcm, gcm->X); + gcm->buflen = 0; + gcm->totlen += 128; + } + } + + return CRYPT_OK; +} +#endif + + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file gcm_reset.c + GCM implementation, reset a used state so it can accept IV data, by Tom St Denis +*/ + + +#ifdef LTC_GCM_MODE + +/** + Reset a GCM state to as if you just called gcm_init(). This saves the initialization time. + @param gcm The GCM state to reset + @return CRYPT_OK on success +*/ +int gcm_reset(gcm_state *gcm) +{ + LTC_ARGCHK(gcm != NULL); + + zeromem(gcm->buf, sizeof(gcm->buf)); + zeromem(gcm->X, sizeof(gcm->X)); + gcm->mode = LTC_GCM_MODE_IV; + gcm->ivmode = 0; + gcm->buflen = 0; + gcm->totlen = 0; + gcm->pttotlen = 0; + + return CRYPT_OK; +} + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + + + +/** + @file md5.c + LTC_MD5 hash function by Tom St Denis +*/ + +#ifdef LTC_MD5 + +const struct ltc_hash_descriptor md5_desc = +{ + "md5", + 3, + 16, + 64, + + /* OID */ + { 1, 2, 840, 113549, 2, 5, }, + 6, + + &md5_init, + &md5_process, + &md5_done, + &md5_test, + NULL +}; + +#define F(x,y,z) (z ^ (x & (y ^ z))) +#define G(x,y,z) (y ^ (z & (y ^ x))) +#define H(x,y,z) (x^y^z) +#define I(x,y,z) (y^(x|(~z))) + +#ifdef LTC_SMALL_CODE + +#define FF(a,b,c,d,M,s,t) \ + a = (a + F(b,c,d) + M + t); a = ROL(a, s) + b; + +#define GG(a,b,c,d,M,s,t) \ + a = (a + G(b,c,d) + M + t); a = ROL(a, s) + b; + +#define HH(a,b,c,d,M,s,t) \ + a = (a + H(b,c,d) + M + t); a = ROL(a, s) + b; + +#define II(a,b,c,d,M,s,t) \ + a = (a + I(b,c,d) + M + t); a = ROL(a, s) + b; + +static const unsigned char Worder[64] = { + 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15, + 1,6,11,0,5,10,15,4,9,14,3,8,13,2,7,12, + 5,8,11,14,1,4,7,10,13,0,3,6,9,12,15,2, + 0,7,14,5,12,3,10,1,8,15,6,13,4,11,2,9 +}; + +static const unsigned char Rorder[64] = { + 7,12,17,22,7,12,17,22,7,12,17,22,7,12,17,22, + 5,9,14,20,5,9,14,20,5,9,14,20,5,9,14,20, + 4,11,16,23,4,11,16,23,4,11,16,23,4,11,16,23, + 6,10,15,21,6,10,15,21,6,10,15,21,6,10,15,21 +}; + +static const ulong32 Korder[64] = { +0xd76aa478UL, 0xe8c7b756UL, 0x242070dbUL, 0xc1bdceeeUL, 0xf57c0fafUL, 0x4787c62aUL, 0xa8304613UL, 0xfd469501UL, +0x698098d8UL, 0x8b44f7afUL, 0xffff5bb1UL, 0x895cd7beUL, 0x6b901122UL, 0xfd987193UL, 0xa679438eUL, 0x49b40821UL, +0xf61e2562UL, 0xc040b340UL, 0x265e5a51UL, 0xe9b6c7aaUL, 0xd62f105dUL, 0x02441453UL, 0xd8a1e681UL, 0xe7d3fbc8UL, +0x21e1cde6UL, 0xc33707d6UL, 0xf4d50d87UL, 0x455a14edUL, 0xa9e3e905UL, 0xfcefa3f8UL, 0x676f02d9UL, 0x8d2a4c8aUL, +0xfffa3942UL, 0x8771f681UL, 0x6d9d6122UL, 0xfde5380cUL, 0xa4beea44UL, 0x4bdecfa9UL, 0xf6bb4b60UL, 0xbebfbc70UL, +0x289b7ec6UL, 0xeaa127faUL, 0xd4ef3085UL, 0x04881d05UL, 0xd9d4d039UL, 0xe6db99e5UL, 0x1fa27cf8UL, 0xc4ac5665UL, +0xf4292244UL, 0x432aff97UL, 0xab9423a7UL, 0xfc93a039UL, 0x655b59c3UL, 0x8f0ccc92UL, 0xffeff47dUL, 0x85845dd1UL, +0x6fa87e4fUL, 0xfe2ce6e0UL, 0xa3014314UL, 0x4e0811a1UL, 0xf7537e82UL, 0xbd3af235UL, 0x2ad7d2bbUL, 0xeb86d391UL +}; + +#else + +#define FF(a,b,c,d,M,s,t) \ + a = (a + F(b,c,d) + M + t); a = ROLc(a, s) + b; + +#define GG(a,b,c,d,M,s,t) \ + a = (a + G(b,c,d) + M + t); a = ROLc(a, s) + b; + +#define HH(a,b,c,d,M,s,t) \ + a = (a + H(b,c,d) + M + t); a = ROLc(a, s) + b; + +#define II(a,b,c,d,M,s,t) \ + a = (a + I(b,c,d) + M + t); a = ROLc(a, s) + b; + + +#endif + +#ifdef LTC_CLEAN_STACK +static int _md5_compress(hash_state *md, unsigned char *buf) +#else +static int md5_compress(hash_state *md, unsigned char *buf) +#endif +{ + ulong32 i, W[16], a, b, c, d; +#ifdef LTC_SMALL_CODE + ulong32 t; +#endif + + /* copy the state into 512-bits into W[0..15] */ + for (i = 0; i < 16; i++) { + LOAD32L(W[i], buf + (4*i)); + } + + /* copy state */ + a = md->md5.state[0]; + b = md->md5.state[1]; + c = md->md5.state[2]; + d = md->md5.state[3]; + +#ifdef LTC_SMALL_CODE + for (i = 0; i < 16; ++i) { + FF(a,b,c,d,W[Worder[i]],Rorder[i],Korder[i]); + t = d; d = c; c = b; b = a; a = t; + } + + for (; i < 32; ++i) { + GG(a,b,c,d,W[Worder[i]],Rorder[i],Korder[i]); + t = d; d = c; c = b; b = a; a = t; + } + + for (; i < 48; ++i) { + HH(a,b,c,d,W[Worder[i]],Rorder[i],Korder[i]); + t = d; d = c; c = b; b = a; a = t; + } + + for (; i < 64; ++i) { + II(a,b,c,d,W[Worder[i]],Rorder[i],Korder[i]); + t = d; d = c; c = b; b = a; a = t; + } + +#else + FF(a,b,c,d,W[0],7,0xd76aa478UL) + FF(d,a,b,c,W[1],12,0xe8c7b756UL) + FF(c,d,a,b,W[2],17,0x242070dbUL) + FF(b,c,d,a,W[3],22,0xc1bdceeeUL) + FF(a,b,c,d,W[4],7,0xf57c0fafUL) + FF(d,a,b,c,W[5],12,0x4787c62aUL) + FF(c,d,a,b,W[6],17,0xa8304613UL) + FF(b,c,d,a,W[7],22,0xfd469501UL) + FF(a,b,c,d,W[8],7,0x698098d8UL) + FF(d,a,b,c,W[9],12,0x8b44f7afUL) + FF(c,d,a,b,W[10],17,0xffff5bb1UL) + FF(b,c,d,a,W[11],22,0x895cd7beUL) + FF(a,b,c,d,W[12],7,0x6b901122UL) + FF(d,a,b,c,W[13],12,0xfd987193UL) + FF(c,d,a,b,W[14],17,0xa679438eUL) + FF(b,c,d,a,W[15],22,0x49b40821UL) + GG(a,b,c,d,W[1],5,0xf61e2562UL) + GG(d,a,b,c,W[6],9,0xc040b340UL) + GG(c,d,a,b,W[11],14,0x265e5a51UL) + GG(b,c,d,a,W[0],20,0xe9b6c7aaUL) + GG(a,b,c,d,W[5],5,0xd62f105dUL) + GG(d,a,b,c,W[10],9,0x02441453UL) + GG(c,d,a,b,W[15],14,0xd8a1e681UL) + GG(b,c,d,a,W[4],20,0xe7d3fbc8UL) + GG(a,b,c,d,W[9],5,0x21e1cde6UL) + GG(d,a,b,c,W[14],9,0xc33707d6UL) + GG(c,d,a,b,W[3],14,0xf4d50d87UL) + GG(b,c,d,a,W[8],20,0x455a14edUL) + GG(a,b,c,d,W[13],5,0xa9e3e905UL) + GG(d,a,b,c,W[2],9,0xfcefa3f8UL) + GG(c,d,a,b,W[7],14,0x676f02d9UL) + GG(b,c,d,a,W[12],20,0x8d2a4c8aUL) + HH(a,b,c,d,W[5],4,0xfffa3942UL) + HH(d,a,b,c,W[8],11,0x8771f681UL) + HH(c,d,a,b,W[11],16,0x6d9d6122UL) + HH(b,c,d,a,W[14],23,0xfde5380cUL) + HH(a,b,c,d,W[1],4,0xa4beea44UL) + HH(d,a,b,c,W[4],11,0x4bdecfa9UL) + HH(c,d,a,b,W[7],16,0xf6bb4b60UL) + HH(b,c,d,a,W[10],23,0xbebfbc70UL) + HH(a,b,c,d,W[13],4,0x289b7ec6UL) + HH(d,a,b,c,W[0],11,0xeaa127faUL) + HH(c,d,a,b,W[3],16,0xd4ef3085UL) + HH(b,c,d,a,W[6],23,0x04881d05UL) + HH(a,b,c,d,W[9],4,0xd9d4d039UL) + HH(d,a,b,c,W[12],11,0xe6db99e5UL) + HH(c,d,a,b,W[15],16,0x1fa27cf8UL) + HH(b,c,d,a,W[2],23,0xc4ac5665UL) + II(a,b,c,d,W[0],6,0xf4292244UL) + II(d,a,b,c,W[7],10,0x432aff97UL) + II(c,d,a,b,W[14],15,0xab9423a7UL) + II(b,c,d,a,W[5],21,0xfc93a039UL) + II(a,b,c,d,W[12],6,0x655b59c3UL) + II(d,a,b,c,W[3],10,0x8f0ccc92UL) + II(c,d,a,b,W[10],15,0xffeff47dUL) + II(b,c,d,a,W[1],21,0x85845dd1UL) + II(a,b,c,d,W[8],6,0x6fa87e4fUL) + II(d,a,b,c,W[15],10,0xfe2ce6e0UL) + II(c,d,a,b,W[6],15,0xa3014314UL) + II(b,c,d,a,W[13],21,0x4e0811a1UL) + II(a,b,c,d,W[4],6,0xf7537e82UL) + II(d,a,b,c,W[11],10,0xbd3af235UL) + II(c,d,a,b,W[2],15,0x2ad7d2bbUL) + II(b,c,d,a,W[9],21,0xeb86d391UL) +#endif + + md->md5.state[0] = md->md5.state[0] + a; + md->md5.state[1] = md->md5.state[1] + b; + md->md5.state[2] = md->md5.state[2] + c; + md->md5.state[3] = md->md5.state[3] + d; + + return CRYPT_OK; +} + +#ifdef LTC_CLEAN_STACK +static int md5_compress(hash_state *md, unsigned char *buf) +{ + int err; + err = _md5_compress(md, buf); + burn_stack(sizeof(ulong32) * 21); + return err; +} +#endif + +/** + Initialize the hash state + @param md The hash state you wish to initialize + @return CRYPT_OK if successful +*/ +int md5_init(hash_state * md) +{ + LTC_ARGCHK(md != NULL); + md->md5.state[0] = 0x67452301UL; + md->md5.state[1] = 0xefcdab89UL; + md->md5.state[2] = 0x98badcfeUL; + md->md5.state[3] = 0x10325476UL; + md->md5.curlen = 0; + md->md5.length = 0; + return CRYPT_OK; +} + +/** + Process a block of memory though the hash + @param md The hash state + @param in The data to hash + @param inlen The length of the data (octets) + @return CRYPT_OK if successful +*/ +HASH_PROCESS(md5_process, md5_compress, md5, 64) + +/** + Terminate the hash to get the digest + @param md The hash state + @param out [out] The destination of the hash (16 bytes) + @return CRYPT_OK if successful +*/ +int md5_done(hash_state * md, unsigned char *out) +{ + int i; + + LTC_ARGCHK(md != NULL); + LTC_ARGCHK(out != NULL); + + if (md->md5.curlen >= sizeof(md->md5.buf)) { + return CRYPT_INVALID_ARG; + } + + + /* increase the length of the message */ + md->md5.length += md->md5.curlen * 8; + + /* append the '1' bit */ + md->md5.buf[md->md5.curlen++] = (unsigned char)0x80; + + /* if the length is currently above 56 bytes we append zeros + * then compress. Then we can fall back to padding zeros and length + * encoding like normal. + */ + if (md->md5.curlen > 56) { + while (md->md5.curlen < 64) { + md->md5.buf[md->md5.curlen++] = (unsigned char)0; + } + md5_compress(md, md->md5.buf); + md->md5.curlen = 0; + } + + /* pad upto 56 bytes of zeroes */ + while (md->md5.curlen < 56) { + md->md5.buf[md->md5.curlen++] = (unsigned char)0; + } + + /* store length */ + STORE64L(md->md5.length, md->md5.buf+56); + md5_compress(md, md->md5.buf); + + /* copy output */ + for (i = 0; i < 4; i++) { + STORE32L(md->md5.state[i], out+(4*i)); + } +#ifdef LTC_CLEAN_STACK + zeromem(md, sizeof(hash_state)); +#endif + return CRYPT_OK; +} + +/** + Self-test the hash + @return CRYPT_OK if successful, CRYPT_NOP if self-tests have been disabled +*/ +int md5_test(void) +{ + #ifndef LTC_TEST + return CRYPT_NOP; + #else + static const struct { + char *msg; + unsigned char hash[16]; + } tests[] = { + { "", + { 0xd4, 0x1d, 0x8c, 0xd9, 0x8f, 0x00, 0xb2, 0x04, + 0xe9, 0x80, 0x09, 0x98, 0xec, 0xf8, 0x42, 0x7e } }, + { "a", + {0x0c, 0xc1, 0x75, 0xb9, 0xc0, 0xf1, 0xb6, 0xa8, + 0x31, 0xc3, 0x99, 0xe2, 0x69, 0x77, 0x26, 0x61 } }, + { "abc", + { 0x90, 0x01, 0x50, 0x98, 0x3c, 0xd2, 0x4f, 0xb0, + 0xd6, 0x96, 0x3f, 0x7d, 0x28, 0xe1, 0x7f, 0x72 } }, + { "message digest", + { 0xf9, 0x6b, 0x69, 0x7d, 0x7c, 0xb7, 0x93, 0x8d, + 0x52, 0x5a, 0x2f, 0x31, 0xaa, 0xf1, 0x61, 0xd0 } }, + { "abcdefghijklmnopqrstuvwxyz", + { 0xc3, 0xfc, 0xd3, 0xd7, 0x61, 0x92, 0xe4, 0x00, + 0x7d, 0xfb, 0x49, 0x6c, 0xca, 0x67, 0xe1, 0x3b } }, + { "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789", + { 0xd1, 0x74, 0xab, 0x98, 0xd2, 0x77, 0xd9, 0xf5, + 0xa5, 0x61, 0x1c, 0x2c, 0x9f, 0x41, 0x9d, 0x9f } }, + { "12345678901234567890123456789012345678901234567890123456789012345678901234567890", + { 0x57, 0xed, 0xf4, 0xa2, 0x2b, 0xe3, 0xc9, 0x55, + 0xac, 0x49, 0xda, 0x2e, 0x21, 0x07, 0xb6, 0x7a } }, + { NULL, { 0 } } + }; + + int i; + unsigned char tmp[16]; + hash_state md; + + for (i = 0; tests[i].msg != NULL; i++) { + md5_init(&md); + md5_process(&md, (unsigned char *)tests[i].msg, (unsigned long)strlen(tests[i].msg)); + md5_done(&md, tmp); + if (XMEMCMP(tmp, tests[i].hash, 16) != 0) { + return CRYPT_FAIL_TESTVECTOR; + } + } + return CRYPT_OK; + #endif +} + +#endif + +/* $Source$ */ +/* $Revision$ */ +/* $Date$ */ + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file ctr_encrypt.c + CTR implementation, encrypt data, Tom St Denis +*/ + + +#ifdef LTC_CTR_MODE + +/** + CTR encrypt + @param pt Plaintext + @param ct [out] Ciphertext + @param len Length of plaintext (octets) + @param ctr CTR state + @return CRYPT_OK if successful +*/ +int ctr_encrypt(const unsigned char *pt, unsigned char *ct, unsigned long len, symmetric_CTR *ctr) +{ + int x, err; + + LTC_ARGCHK(pt != NULL); + LTC_ARGCHK(ct != NULL); + LTC_ARGCHK(ctr != NULL); + + if ((err = cipher_is_valid(ctr->cipher)) != CRYPT_OK) { + return err; + } + + /* is blocklen/padlen valid? */ + if (ctr->blocklen < 1 || ctr->blocklen > (int)sizeof(ctr->ctr) || + ctr->padlen < 0 || ctr->padlen > (int)sizeof(ctr->pad)) { + return CRYPT_INVALID_ARG; + } + +#ifdef LTC_FAST + if (ctr->blocklen % sizeof(LTC_FAST_TYPE)) { + return CRYPT_INVALID_ARG; + } +#endif + + /* handle acceleration only if pad is empty, accelerator is present and length is >= a block size */ + if ((ctr->padlen == ctr->blocklen) && cipher_descriptor[ctr->cipher].accel_ctr_encrypt != NULL && (len >= (unsigned long)ctr->blocklen)) { + if ((err = cipher_descriptor[ctr->cipher].accel_ctr_encrypt(pt, ct, len/ctr->blocklen, ctr->ctr, ctr->mode, &ctr->key)) != CRYPT_OK) { + return err; + } + len %= ctr->blocklen; + } + + while (len) { + /* is the pad empty? */ + if (ctr->padlen == ctr->blocklen) { + /* increment counter */ + if (ctr->mode == CTR_COUNTER_LITTLE_ENDIAN) { + /* little-endian */ + for (x = 0; x < ctr->ctrlen; x++) { + ctr->ctr[x] = (ctr->ctr[x] + (unsigned char)1) & (unsigned char)255; + if (ctr->ctr[x] != (unsigned char)0) { + break; + } + } + } else { + /* big-endian */ + for (x = ctr->blocklen-1; x >= ctr->ctrlen; x--) { + ctr->ctr[x] = (ctr->ctr[x] + (unsigned char)1) & (unsigned char)255; + if (ctr->ctr[x] != (unsigned char)0) { + break; + } + } + } + + /* encrypt it */ + if ((err = cipher_descriptor[ctr->cipher].ecb_encrypt(ctr->ctr, ctr->pad, &ctr->key)) != CRYPT_OK) { + return err; + } + ctr->padlen = 0; + } +#ifdef LTC_FAST + if (ctr->padlen == 0 && len >= (unsigned long)ctr->blocklen) { + for (x = 0; x < ctr->blocklen; x += sizeof(LTC_FAST_TYPE)) { + *((LTC_FAST_TYPE*)((unsigned char *)ct + x)) = *((LTC_FAST_TYPE*)((unsigned char *)pt + x)) ^ + *((LTC_FAST_TYPE*)((unsigned char *)ctr->pad + x)); + } + pt += ctr->blocklen; + ct += ctr->blocklen; + len -= ctr->blocklen; + ctr->padlen = ctr->blocklen; + continue; + } +#endif + *ct++ = *pt++ ^ ctr->pad[ctr->padlen++]; + --len; + } + return CRYPT_OK; +} + +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/modes/ctr/ctr_encrypt.c,v $ */ +/* $Revision: 1.22 $ */ +/* $Date: 2007/02/22 20:26:05 $ */ + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file ctr_done.c + CTR implementation, finish chain, Tom St Denis +*/ + +#ifdef LTC_CTR_MODE + +/** Terminate the chain + @param ctr The CTR chain to terminate + @return CRYPT_OK on success +*/ +int ctr_done(symmetric_CTR *ctr) +{ + int err; + LTC_ARGCHK(ctr != NULL); + + if ((err = cipher_is_valid(ctr->cipher)) != CRYPT_OK) { + return err; + } + cipher_descriptor[ctr->cipher].done(&ctr->key); + return CRYPT_OK; +} + + + +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/modes/ctr/ctr_done.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file ctr_decrypt.c + CTR implementation, decrypt data, Tom St Denis +*/ + +#ifdef LTC_CTR_MODE + +/** + CTR decrypt + @param ct Ciphertext + @param pt [out] Plaintext + @param len Length of ciphertext (octets) + @param ctr CTR state + @return CRYPT_OK if successful +*/ +int ctr_decrypt(const unsigned char *ct, unsigned char *pt, unsigned long len, symmetric_CTR *ctr) +{ + LTC_ARGCHK(pt != NULL); + LTC_ARGCHK(ct != NULL); + LTC_ARGCHK(ctr != NULL); + + return ctr_encrypt(ct, pt, len, ctr); +} + +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/modes/ctr/ctr_decrypt.c,v $ */ +/* $Revision: 1.6 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file ctr_start.c + CTR implementation, start chain, Tom St Denis +*/ + + +#ifdef LTC_CTR_MODE + +/** + Initialize a CTR context + @param cipher The index of the cipher desired + @param IV The initial vector + @param key The secret key + @param keylen The length of the secret key (octets) + @param num_rounds Number of rounds in the cipher desired (0 for default) + @param ctr_mode The counter mode (CTR_COUNTER_LITTLE_ENDIAN or CTR_COUNTER_BIG_ENDIAN) + @param ctr The CTR state to initialize + @return CRYPT_OK if successful +*/ +int ctr_start( int cipher, + const unsigned char *IV, + const unsigned char *key, int keylen, + int num_rounds, int ctr_mode, + symmetric_CTR *ctr) +{ + int x, err; + + LTC_ARGCHK(IV != NULL); + LTC_ARGCHK(key != NULL); + LTC_ARGCHK(ctr != NULL); + + /* bad param? */ + if ((err = cipher_is_valid(cipher)) != CRYPT_OK) { + return err; + } + + /* ctrlen == counter width */ + ctr->ctrlen = (ctr_mode & 255) ? (ctr_mode & 255) : cipher_descriptor[cipher].block_length; + if (ctr->ctrlen > cipher_descriptor[cipher].block_length) { + return CRYPT_INVALID_ARG; + } + + if ((ctr_mode & 0x1000) == CTR_COUNTER_BIG_ENDIAN) { + ctr->ctrlen = cipher_descriptor[cipher].block_length - ctr->ctrlen; + } + + /* setup cipher */ + if ((err = cipher_descriptor[cipher].setup(key, keylen, num_rounds, &ctr->key)) != CRYPT_OK) { + return err; + } + + /* copy ctr */ + ctr->blocklen = cipher_descriptor[cipher].block_length; + ctr->cipher = cipher; + ctr->padlen = 0; + ctr->mode = ctr_mode & 0x1000; + for (x = 0; x < ctr->blocklen; x++) { + ctr->ctr[x] = IV[x]; + } + + if (ctr_mode & LTC_CTR_RFC3686) { + /* increment the IV as per RFC 3686 */ + if (ctr->mode == CTR_COUNTER_LITTLE_ENDIAN) { + /* little-endian */ + for (x = 0; x < ctr->ctrlen; x++) { + ctr->ctr[x] = (ctr->ctr[x] + (unsigned char)1) & (unsigned char)255; + if (ctr->ctr[x] != (unsigned char)0) { + break; + } + } + } else { + /* big-endian */ + for (x = ctr->blocklen-1; x >= ctr->ctrlen; x--) { + ctr->ctr[x] = (ctr->ctr[x] + (unsigned char)1) & (unsigned char)255; + if (ctr->ctr[x] != (unsigned char)0) { + break; + } + } + } + } + + return cipher_descriptor[ctr->cipher].ecb_encrypt(ctr->ctr, ctr->pad, &ctr->key); +} + +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/modes/ctr/ctr_start.c,v $ */ +/* $Revision: 1.15 $ */ +/* $Date: 2007/02/23 14:18:37 $ */ + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file ctr_setiv.c + CTR implementation, set IV, Tom St Denis +*/ + +#ifdef LTC_CTR_MODE + +/** + Set an initial vector + @param IV The initial vector + @param len The length of the vector (in octets) + @param ctr The CTR state + @return CRYPT_OK if successful +*/ +int ctr_setiv(const unsigned char *IV, unsigned long len, symmetric_CTR *ctr) +{ + int err; + + LTC_ARGCHK(IV != NULL); + LTC_ARGCHK(ctr != NULL); + + /* bad param? */ + if ((err = cipher_is_valid(ctr->cipher)) != CRYPT_OK) { + return err; + } + + if (len != (unsigned long)ctr->blocklen) { + return CRYPT_INVALID_ARG; + } + + /* set IV */ + XMEMCPY(ctr->ctr, IV, len); + + /* force next block */ + ctr->padlen = 0; + return cipher_descriptor[ctr->cipher].ecb_encrypt(IV, ctr->pad, &ctr->key); +} + +#endif + + +/* $Source: /cvs/libtom/libtomcrypt/src/modes/ctr/ctr_setiv.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ + +/* LibTomCrypt, modular cryptographic library -- Tom St Denis + * + * LibTomCrypt is a library that provides various cryptographic + * algorithms in a highly modular and flexible manner. + * + * The library is free for all purposes without any express + * guarantee it works. + * + * Tom St Denis, tomstdenis@gmail.com, http://libtom.org + */ + +/** + @file ctr_getiv.c + CTR implementation, get IV, Tom St Denis +*/ + +#ifdef LTC_CTR_MODE + +/** + Get the current initial vector + @param IV [out] The destination of the initial vector + @param len [in/out] The max size and resulting size of the initial vector + @param ctr The CTR state + @return CRYPT_OK if successful +*/ +int ctr_getiv(unsigned char *IV, unsigned long *len, symmetric_CTR *ctr) +{ + LTC_ARGCHK(IV != NULL); + LTC_ARGCHK(len != NULL); + LTC_ARGCHK(ctr != NULL); + if ((unsigned long)ctr->blocklen > *len) { + *len = ctr->blocklen; + return CRYPT_BUFFER_OVERFLOW; + } + XMEMCPY(IV, ctr->ctr, ctr->blocklen); + *len = ctr->blocklen; + + return CRYPT_OK; +} + +#endif + +/* $Source: /cvs/libtom/libtomcrypt/src/modes/ctr/ctr_getiv.c,v $ */ +/* $Revision: 1.7 $ */ +/* $Date: 2006/12/28 01:27:24 $ */ diff --git a/src/tlse/tlse.c b/src/tlse/tlse.c new file mode 100644 index 0000000..957053d --- /dev/null +++ b/src/tlse/tlse.c @@ -0,0 +1,12374 @@ +/******************************************************************************** + Copyright (c) 2016-2024, Eduard Suica + All rights reserved. + + Redistribution and use in source and binary forms, with or without modification, + are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, this + list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright notice, this + list of conditions and the following disclaimer in the documentation and/or other + materials provided with the distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND + ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. + IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT + NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR + PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, + WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. + ********************************************************************************/ +#ifndef TLSE_C +#define TLSE_C + +#include +#include +#include +#include +#include +#ifdef _WIN32 +#ifdef SSL_COMPATIBLE_INTERFACE +#include +#endif +#include +#include +#ifndef strcasecmp + #define strcasecmp stricmp +#endif +#else +// hton* and ntoh* functions +#include +#include +#include +#endif + +#ifdef TLS_AMALGAMATION +#ifdef I +#pragma push_macro("I") +#define TLS_I_MACRO +#undef I +#endif +#include "libtomcrypt.c" +#ifdef TLS_I_MACRO +#pragma pop_macro("I") +#undef TLS_I_MACRO +#endif +#else +#include +#endif + +#if (CRYPT <= 0x0117) + #define LTC_PKCS_1_EMSA LTC_LTC_PKCS_1_EMSA + #define LTC_PKCS_1_V1_5 LTC_LTC_PKCS_1_V1_5 + #define LTC_PKCS_1_PSS LTC_LTC_PKCS_1_PSS +#endif + +#ifdef WITH_KTLS + #include + #include + #include + // should get /usr/include/linux/tls.h (linux headers) + // rename it to ktls.h and add it to your project + #include "ktls.h" + // or just include tls.h instead of ktls.h + // #include "linux/tls.h" +#endif + +#include "tlse.h" +#ifdef TLS_CURVE25519 + #include "curve25519.c" +#endif +// using ChaCha20 implementation by D. J. Bernstein + +#ifndef TLS_FORWARD_SECRECY +#undef TLS_ECDSA_SUPPORTED +#endif + +#ifndef TLS_ECDSA_SUPPORTED +// disable client ECDSA if not supported +#undef TLS_CLIENT_ECDSA +#endif + +#define TLS_DH_DEFAULT_P "87A8E61DB4B6663CFFBBD19C651959998CEEF608660DD0F25D2CEED4435E3B00E00DF8F1D61957D4FAF7DF4561B2AA3016C3D91134096FAA3BF4296D830E9A7C209E0C6497517ABD5A8A9D306BCF67ED91F9E6725B4758C022E0B1EF4275BF7B6C5BFC11D45F9088B941F54EB1E59BB8BC39A0BF12307F5C4FDB70C581B23F76B63ACAE1CAA6B7902D52526735488A0EF13C6D9A51BFA4AB3AD8347796524D8EF6A167B5A41825D967E144E5140564251CCACB83E6B486F6B3CA3F7971506026C0B857F689962856DED4010ABD0BE621C3A3960A54E710C375F26375D7014103A4B54330C198AF126116D2276E11715F693877FAD7EF09CADB094AE91E1A1597" +#define TLS_DH_DEFAULT_G "3FB32C9B73134D0B2E77506660EDBD484CA7B18F21EF205407F4793A1A0BA12510DBC15077BE463FFF4FED4AAC0BB555BE3A6C1B0C6B47B1BC3773BF7E8C6F62901228F8C28CBB18A55AE31341000A650196F931C77A57F2DDF463E5E9EC144B777DE62AAAB8A8628AC376D282D6ED3864E67982428EBC831D14348F6F2F9193B5045AF2767164E1DFC967C1FB3F2E55A4BD1BFFE83B9C80D052B985D182EA0ADB2A3B7313D3FE14C8484B1E052588B9B7D2BBD2DF016199ECD06E1557CD0915B3353BBB64E0EC377FD028370DF92B52C7891428CDC67EB6184B523D1DB246C32F63078490F00EF8D647D148D47954515E2327CFEF98C582664B4C0F6CC41659" +#define TLS_DHE_KEY_SIZE 2048 + +// you should never use weak DH groups (1024 bits) +// but if you have old devices (like grandstream ip phones) +// that can't handle 2048bit DHE, uncomment next lines +// and define TLS_WEAK_DH_LEGACY_DEVICES +// #ifdef TLS_WEAK_DH_LEGACY_DEVICES +// #define TLS_DH_DEFAULT_P "B10B8F96A080E01DDE92DE5EAE5D54EC52C99FBCFB06A3C69A6A9DCA52D23B616073E28675A23D189838EF1E2EE652C013ECB4AEA906112324975C3CD49B83BFACCBDD7D90C4BD7098488E9C219A73724EFFD6FAE5644738FAA31A4FF55BCCC0A151AF5F0DC8B4BD45BF37DF365C1A65E68CFDA76D4DA708DF1FB2BC2E4A4371" +// #define TLS_DH_DEFAULT_G "A4D1CBD5C3FD34126765A442EFB99905F8104DD258AC507FD6406CFF14266D31266FEA1E5C41564B777E690F5504F213160217B4B01B886A5E91547F9E2749F4D7FBD7D3B9A92EE1909D0D2263F80A76A6A24C087A091F531DBF0A0169B6A28AD662A4D18E73AFA32D779D5918D08BC8858F4DCEF97C2A24855E6EEB22B3B2E5" +// #define TLS_DHE_KEY_SIZE 1024 +// #endif + +#ifndef TLS_MALLOC + #define TLS_MALLOC(size) malloc(size) +#endif +#ifndef TLS_REALLOC + #define TLS_REALLOC(ptr, size) realloc(ptr, size) +#endif +#ifndef TLS_FREE + #define TLS_FREE(ptr) if (ptr) free(ptr) +#endif + +#define TLS_ERROR(err, statement) if (err) statement; + +#ifdef DEBUG +#define DEBUG_PRINT(...) fprintf(stderr, __VA_ARGS__) +#define DEBUG_DUMP_HEX(buf, len) {if (buf) { int _i_; for (_i_ = 0; _i_ < len; _i_++) { DEBUG_PRINT("%02X ", (unsigned int)(buf)[_i_]); } } else { fprintf(stderr, "(null)"); } } +#define DEBUG_INDEX(fields) print_index(fields) +#define DEBUG_DUMP(buf, length) fwrite(buf, 1, length, stderr); +#define DEBUG_DUMP_HEX_LABEL(title, buf, len) {fprintf(stderr, "%s (%i): ", title, (int)len); DEBUG_DUMP_HEX(buf, len); fprintf(stderr, "\n");} +#else +#define DEBUG_PRINT(...) { } +#define DEBUG_DUMP_HEX(buf, len) { } +#define DEBUG_INDEX(fields) { } +#define DEBUG_DUMP(buf, length) { } +#define DEBUG_DUMP_HEX_LABEL(title, buf, len) { } +#endif + +#ifndef htonll +#define htonll(x) ((1==htonl(1)) ? (x) : ((uint64_t)htonl((x) & 0xFFFFFFFF) << 32) | htonl((x) >> 32)) +#endif + +#ifndef ntohll +#define ntohll(x) ((1==ntohl(1)) ? (x) : ((uint64_t)ntohl((x) & 0xFFFFFFFF) << 32) | ntohl((x) >> 32)) +#endif + +#define TLS_CHANGE_CIPHER 0x14 +#define TLS_ALERT 0x15 +#define TLS_HANDSHAKE 0x16 +#define TLS_APPLICATION_DATA 0x17 + +#define TLS_SERIALIZED_OBJECT 0xFE + +#define TLS_CLIENT_HELLO_MINSIZE 41 +#define TLS_CLIENT_RANDOM_SIZE 32 +#define TLS_SERVER_RANDOM_SIZE 32 +#define TLS_MAX_SESSION_ID 32 +#define TLS_SHA256_MAC_SIZE 32 +#define TLS_SHA1_MAC_SIZE 20 +#define TLS_SHA384_MAC_SIZE 48 +#define TLS_MAX_MAC_SIZE TLS_SHA384_MAC_SIZE + // 160 +#define TLS_MAX_KEY_EXPANSION_SIZE 192 +// 512bits (sha256) = 64 bytes +#define TLS_MAX_HASH_LEN 64 +#define TLS_AES_IV_LENGTH 16 +#define TLS_AES_BLOCK_SIZE 16 +#define TLS_AES_GCM_IV_LENGTH 4 +#define TLS_13_AES_GCM_IV_LENGTH 12 +#define TLS_GCM_TAG_LEN 16 +#define TLS_MAX_TAG_LEN 16 +#define TLS_MIN_FINISHED_OPAQUE_LEN 12 + +#define TLS_BLOB_INCREMENT 0xFFF +#define TLS_ASN1_MAXLEVEL 0xFF + +#define DTLS_COOKIE_SIZE 32 +#define DTLS_MAX_FRAGMENT_SIZE 0x40000 + +#define TLS_MAX_SHA_SIZE 48 +// 16(md5) + 20(sha1) +#define TLS_V11_HASH_SIZE 36 +#define TLS_MAX_HASH_SIZE TLS_MAX_SHA_SIZE +// 16(md5) + 20(sha1) +#define TLS_MAX_RSA_KEY 2048 + +#define TLS_MAXTLS_APP_SIZE 0x4000 +// max 1 second sleep +#define TLS_MAX_ERROR_SLEEP_uS 1000000 +// max 5 seconds context sleep +#define TLS_MAX_ERROR_IDLE_S 5 + +#define TLS_V13_MAX_KEY_SIZE 32 +#define TLS_V13_MAX_IV_SIZE 12 + +#define VERSION_SUPPORTED(version, err) if ((version != TLS_V13) && (version != TLS_V12) && (version != TLS_V11) && (version != TLS_V10) && (version != DTLS_V13) && (version != DTLS_V12) && (version != DTLS_V10)) { if ((version == SSL_V30) && (context->connection_status == 0)) { version = TLS_V12; } else { DEBUG_PRINT("UNSUPPORTED TLS VERSION %x\n", (int)version); return err;} } +#define CHECK_SIZE(size, buf_size, err) if (((int)(size) > (int)(buf_size)) || ((int)(buf_size) < 0)) { DEBUG_PRINT("[EXPECTED AT LEAST %i IN BUFFER OF SIZE %i]\n", (int)(size), (int)(buf_size)); return err; } +#define TLS_IMPORT_CHECK_SIZE(buf_pos, size, buf_size) if (((int)size > (int)buf_size - buf_pos) || ((int)buf_pos > (int)buf_size)) { DEBUG_PRINT("IMPORT ELEMENT SIZE ERROR\n"); tls_destroy_context(context); return NULL; } +#define CHECK_HANDSHAKE_STATE(context, n, limit) { if (context->hs_messages[n] >= limit) { if (context->dtls) { DEBUG_PRINT("* REPEATED MESSAGE, RE-HASHING\n"); _private_dtls_rehash(context, type); context->hs_messages[n]++;} else { DEBUG_PRINT("* UNEXPECTED MESSAGE (%i)\n", (int)n); payload_res = TLS_UNEXPECTED_MESSAGE; break; } } context->hs_messages[n]++; } +#define TLS_24_BIT(buf, index, val) { unsigned int u_val = (unsigned int)val; buf[index] = u_val / 0x10000; u_val %= 0x10000; buf[index + 1] = u_val / 0x100; u_val %= 0x100; buf[index + 2] = u_val; } + +#if CRYPT >= 0x0118 + #define TLS_TOMCRYPT_PRIVATE_DP(key) (&((key)->dp)) + #define TLS_TOMCRYPT_PRIVATE_SET_INDEX(key, k_idx) +#else + #define TLS_TOMCRYPT_PRIVATE_DP(key) ((key)->dp) + #define TLS_TOMCRYPT_PRIVATE_SET_INDEX(key, k_idx) key->idx = k_idx +#endif + +#ifdef TLS_WITH_CHACHA20_POLY1305 +#define TLS_CHACHA20_IV_LENGTH 12 + +// ChaCha20 implementation by D. J. Bernstein +// Public domain. + +#define CHACHA_MINKEYLEN 16 +#define CHACHA_NONCELEN 8 +#define CHACHA_NONCELEN_96 12 +#define CHACHA_CTRLEN 8 +#define CHACHA_CTRLEN_96 4 +#define CHACHA_STATELEN (CHACHA_NONCELEN+CHACHA_CTRLEN) +#define CHACHA_BLOCKLEN 64 + +#define POLY1305_MAX_AAD 32 +#define POLY1305_KEYLEN 32 +#define POLY1305_TAGLEN 16 + +#define u_int unsigned int +#define uint8_t unsigned char +#define u_char unsigned char +#ifndef NULL +#define NULL (void *)0 +#endif + +#if (CRYPT >= 0x0117) && (0) + // to do: use ltc chacha/poly1305 implementation (working on big-endian machines) + #define chacha_ctx chacha20poly1305_state + #define poly1305_context poly1305_state + + #define _private_tls_poly1305_init(ctx, key, len) poly1305_init(ctx, key, len) + #define _private_tls_poly1305_update(ctx, in, len) poly1305_process(ctx, in, len) + #define _private_tls_poly1305_finish(ctx, mac) poly1305_done(ctx, mac, 16) +#else +struct chacha_ctx { + u_int input[16]; + uint8_t ks[CHACHA_BLOCKLEN]; + uint8_t unused; +}; + +static inline void chacha_keysetup(struct chacha_ctx *x, const u_char *k, u_int kbits); +static inline void chacha_ivsetup(struct chacha_ctx *x, const u_char *iv, const u_char *ctr); +static inline void chacha_ivsetup_96bitnonce(struct chacha_ctx *x, const u_char *iv, const u_char *ctr); +static inline void chacha_encrypt_bytes(struct chacha_ctx *x, const u_char *m, u_char *c, u_int bytes); +static inline int poly1305_generate_key(unsigned char *key256, unsigned char *nonce, unsigned int noncelen, unsigned char *poly_key, unsigned int counter); + +#define poly1305_block_size 16 +#define poly1305_context poly1305_state_internal_t + +//========== ChaCha20 from D. J. Bernstein ========= // +// Source available at https://cr.yp.to/chacha.html // + +typedef unsigned char u8; +typedef unsigned int u32; + +typedef struct chacha_ctx chacha_ctx; + +#define U8C(v) (v##U) +#define U32C(v) (v##U) + +#define U8V(v) ((u8)(v) & U8C(0xFF)) +#define U32V(v) ((u32)(v) & U32C(0xFFFFFFFF)) + +#define ROTL32(v, n) \ + (U32V((v) << (n)) | ((v) >> (32 - (n)))) + +#define _private_tls_U8TO32_LITTLE(p) \ + (((u32)((p)[0])) | \ + ((u32)((p)[1]) << 8) | \ + ((u32)((p)[2]) << 16) | \ + ((u32)((p)[3]) << 24)) + +#define _private_tls_U32TO8_LITTLE(p, v) \ + do { \ + (p)[0] = U8V((v)); \ + (p)[1] = U8V((v) >> 8); \ + (p)[2] = U8V((v) >> 16); \ + (p)[3] = U8V((v) >> 24); \ + } while (0) + +#define ROTATE(v,c) (ROTL32(v,c)) +#define XOR(v,w) ((v) ^ (w)) +#define PLUS(v,w) (U32V((v) + (w))) +#define PLUSONE(v) (PLUS((v),1)) + +#define QUARTERROUND(a,b,c,d) \ + a = PLUS(a,b); d = ROTATE(XOR(d,a),16); \ + c = PLUS(c,d); b = ROTATE(XOR(b,c),12); \ + a = PLUS(a,b); d = ROTATE(XOR(d,a), 8); \ + c = PLUS(c,d); b = ROTATE(XOR(b,c), 7); + +static const char sigma[] = "expand 32-byte k"; +static const char tau[] = "expand 16-byte k"; + +static inline void chacha_keysetup(chacha_ctx *x, const u8 *k, u32 kbits) { + const char *constants; + + x->input[4] = _private_tls_U8TO32_LITTLE(k + 0); + x->input[5] = _private_tls_U8TO32_LITTLE(k + 4); + x->input[6] = _private_tls_U8TO32_LITTLE(k + 8); + x->input[7] = _private_tls_U8TO32_LITTLE(k + 12); + if (kbits == 256) { /* recommended */ + k += 16; + constants = sigma; + } else { /* kbits == 128 */ + constants = tau; + } + x->input[8] = _private_tls_U8TO32_LITTLE(k + 0); + x->input[9] = _private_tls_U8TO32_LITTLE(k + 4); + x->input[10] = _private_tls_U8TO32_LITTLE(k + 8); + x->input[11] = _private_tls_U8TO32_LITTLE(k + 12); + x->input[0] = _private_tls_U8TO32_LITTLE(constants + 0); + x->input[1] = _private_tls_U8TO32_LITTLE(constants + 4); + x->input[2] = _private_tls_U8TO32_LITTLE(constants + 8); + x->input[3] = _private_tls_U8TO32_LITTLE(constants + 12); +} + +static inline void chacha_key(chacha_ctx *x, u8 *k) { + _private_tls_U32TO8_LITTLE(k, x->input[4]); + _private_tls_U32TO8_LITTLE(k + 4, x->input[5]); + _private_tls_U32TO8_LITTLE(k + 8, x->input[6]); + _private_tls_U32TO8_LITTLE(k + 12, x->input[7]); + + _private_tls_U32TO8_LITTLE(k + 16, x->input[8]); + _private_tls_U32TO8_LITTLE(k + 20, x->input[9]); + _private_tls_U32TO8_LITTLE(k + 24, x->input[10]); + _private_tls_U32TO8_LITTLE(k + 28, x->input[11]); +} + +static inline void chacha_nonce(chacha_ctx *x, u8 *nonce) { + _private_tls_U32TO8_LITTLE(nonce + 0, x->input[13]); + _private_tls_U32TO8_LITTLE(nonce + 4, x->input[14]); + _private_tls_U32TO8_LITTLE(nonce + 8, x->input[15]); +} + +static inline void chacha_ivsetup(chacha_ctx *x, const u8 *iv, const u8 *counter) { + x->input[12] = counter == NULL ? 0 : _private_tls_U8TO32_LITTLE(counter + 0); + x->input[13] = counter == NULL ? 0 : _private_tls_U8TO32_LITTLE(counter + 4); + if (iv) { + x->input[14] = _private_tls_U8TO32_LITTLE(iv + 0); + x->input[15] = _private_tls_U8TO32_LITTLE(iv + 4); + } +} + +static inline void chacha_ivsetup_96bitnonce(chacha_ctx *x, const u8 *iv, const u8 *counter) { + x->input[12] = counter == NULL ? 0 : _private_tls_U8TO32_LITTLE(counter + 0); + if (iv) { + x->input[13] = _private_tls_U8TO32_LITTLE(iv + 0); + x->input[14] = _private_tls_U8TO32_LITTLE(iv + 4); + x->input[15] = _private_tls_U8TO32_LITTLE(iv + 8); + } +} + +static inline void chacha_ivupdate(chacha_ctx *x, const u8 *iv, const u8 *aad, const u8 *counter) { + x->input[12] = counter == NULL ? 0 : _private_tls_U8TO32_LITTLE(counter + 0); + x->input[13] = _private_tls_U8TO32_LITTLE(iv + 0); + x->input[14] = _private_tls_U8TO32_LITTLE(iv + 4) ^ _private_tls_U8TO32_LITTLE(aad); + x->input[15] = _private_tls_U8TO32_LITTLE(iv + 8) ^ _private_tls_U8TO32_LITTLE(aad + 4); +} + +static inline void chacha_encrypt_bytes(chacha_ctx *x, const u8 *m, u8 *c, u32 bytes) { + u32 x0, x1, x2, x3, x4, x5, x6, x7; + u32 x8, x9, x10, x11, x12, x13, x14, x15; + u32 j0, j1, j2, j3, j4, j5, j6, j7; + u32 j8, j9, j10, j11, j12, j13, j14, j15; + u8 *ctarget = NULL; + u8 tmp[64]; + u_int i; + + if (!bytes) + return; + + j0 = x->input[0]; + j1 = x->input[1]; + j2 = x->input[2]; + j3 = x->input[3]; + j4 = x->input[4]; + j5 = x->input[5]; + j6 = x->input[6]; + j7 = x->input[7]; + j8 = x->input[8]; + j9 = x->input[9]; + j10 = x->input[10]; + j11 = x->input[11]; + j12 = x->input[12]; + j13 = x->input[13]; + j14 = x->input[14]; + j15 = x->input[15]; + + for (;;) { + if (bytes < 64) { + for (i = 0; i < bytes; ++i) + tmp[i] = m[i]; + m = tmp; + ctarget = c; + c = tmp; + } + x0 = j0; + x1 = j1; + x2 = j2; + x3 = j3; + x4 = j4; + x5 = j5; + x6 = j6; + x7 = j7; + x8 = j8; + x9 = j9; + x10 = j10; + x11 = j11; + x12 = j12; + x13 = j13; + x14 = j14; + x15 = j15; + for (i = 20; i > 0; i -= 2) { + QUARTERROUND(x0, x4, x8, x12) + QUARTERROUND(x1, x5, x9, x13) + QUARTERROUND(x2, x6, x10, x14) + QUARTERROUND(x3, x7, x11, x15) + QUARTERROUND(x0, x5, x10, x15) + QUARTERROUND(x1, x6, x11, x12) + QUARTERROUND(x2, x7, x8, x13) + QUARTERROUND(x3, x4, x9, x14) + } + x0 = PLUS(x0, j0); + x1 = PLUS(x1, j1); + x2 = PLUS(x2, j2); + x3 = PLUS(x3, j3); + x4 = PLUS(x4, j4); + x5 = PLUS(x5, j5); + x6 = PLUS(x6, j6); + x7 = PLUS(x7, j7); + x8 = PLUS(x8, j8); + x9 = PLUS(x9, j9); + x10 = PLUS(x10, j10); + x11 = PLUS(x11, j11); + x12 = PLUS(x12, j12); + x13 = PLUS(x13, j13); + x14 = PLUS(x14, j14); + x15 = PLUS(x15, j15); + + if (bytes < 64) { + _private_tls_U32TO8_LITTLE(x->ks + 0, x0); + _private_tls_U32TO8_LITTLE(x->ks + 4, x1); + _private_tls_U32TO8_LITTLE(x->ks + 8, x2); + _private_tls_U32TO8_LITTLE(x->ks + 12, x3); + _private_tls_U32TO8_LITTLE(x->ks + 16, x4); + _private_tls_U32TO8_LITTLE(x->ks + 20, x5); + _private_tls_U32TO8_LITTLE(x->ks + 24, x6); + _private_tls_U32TO8_LITTLE(x->ks + 28, x7); + _private_tls_U32TO8_LITTLE(x->ks + 32, x8); + _private_tls_U32TO8_LITTLE(x->ks + 36, x9); + _private_tls_U32TO8_LITTLE(x->ks + 40, x10); + _private_tls_U32TO8_LITTLE(x->ks + 44, x11); + _private_tls_U32TO8_LITTLE(x->ks + 48, x12); + _private_tls_U32TO8_LITTLE(x->ks + 52, x13); + _private_tls_U32TO8_LITTLE(x->ks + 56, x14); + _private_tls_U32TO8_LITTLE(x->ks + 60, x15); + } + + x0 = XOR(x0, _private_tls_U8TO32_LITTLE(m + 0)); + x1 = XOR(x1, _private_tls_U8TO32_LITTLE(m + 4)); + x2 = XOR(x2, _private_tls_U8TO32_LITTLE(m + 8)); + x3 = XOR(x3, _private_tls_U8TO32_LITTLE(m + 12)); + x4 = XOR(x4, _private_tls_U8TO32_LITTLE(m + 16)); + x5 = XOR(x5, _private_tls_U8TO32_LITTLE(m + 20)); + x6 = XOR(x6, _private_tls_U8TO32_LITTLE(m + 24)); + x7 = XOR(x7, _private_tls_U8TO32_LITTLE(m + 28)); + x8 = XOR(x8, _private_tls_U8TO32_LITTLE(m + 32)); + x9 = XOR(x9, _private_tls_U8TO32_LITTLE(m + 36)); + x10 = XOR(x10, _private_tls_U8TO32_LITTLE(m + 40)); + x11 = XOR(x11, _private_tls_U8TO32_LITTLE(m + 44)); + x12 = XOR(x12, _private_tls_U8TO32_LITTLE(m + 48)); + x13 = XOR(x13, _private_tls_U8TO32_LITTLE(m + 52)); + x14 = XOR(x14, _private_tls_U8TO32_LITTLE(m + 56)); + x15 = XOR(x15, _private_tls_U8TO32_LITTLE(m + 60)); + + j12 = PLUSONE(j12); + if (!j12) { + j13 = PLUSONE(j13); + /* + * Stopping at 2^70 bytes per nonce is the user's + * responsibility. + */ + } + + _private_tls_U32TO8_LITTLE(c + 0, x0); + _private_tls_U32TO8_LITTLE(c + 4, x1); + _private_tls_U32TO8_LITTLE(c + 8, x2); + _private_tls_U32TO8_LITTLE(c + 12, x3); + _private_tls_U32TO8_LITTLE(c + 16, x4); + _private_tls_U32TO8_LITTLE(c + 20, x5); + _private_tls_U32TO8_LITTLE(c + 24, x6); + _private_tls_U32TO8_LITTLE(c + 28, x7); + _private_tls_U32TO8_LITTLE(c + 32, x8); + _private_tls_U32TO8_LITTLE(c + 36, x9); + _private_tls_U32TO8_LITTLE(c + 40, x10); + _private_tls_U32TO8_LITTLE(c + 44, x11); + _private_tls_U32TO8_LITTLE(c + 48, x12); + _private_tls_U32TO8_LITTLE(c + 52, x13); + _private_tls_U32TO8_LITTLE(c + 56, x14); + _private_tls_U32TO8_LITTLE(c + 60, x15); + + if (bytes <= 64) { + if (bytes < 64) { + for (i = 0; i < bytes; ++i) + ctarget[i] = c[i]; + } + x->input[12] = j12; + x->input[13] = j13; + x->unused = 64 - bytes; + return; + } + bytes -= 64; + c += 64; + m += 64; + } +} + +static inline void chacha20_block(chacha_ctx *x, unsigned char *c, u_int len) { + u_int i; + + unsigned int state[16]; + for (i = 0; i < 16; i++) + state[i] = x->input[i]; + for (i = 20; i > 0; i -= 2) { + QUARTERROUND(state[0], state[4], state[8], state[12]) + QUARTERROUND(state[1], state[5], state[9], state[13]) + QUARTERROUND(state[2], state[6], state[10], state[14]) + QUARTERROUND(state[3], state[7], state[11], state[15]) + QUARTERROUND(state[0], state[5], state[10], state[15]) + QUARTERROUND(state[1], state[6], state[11], state[12]) + QUARTERROUND(state[2], state[7], state[8], state[13]) + QUARTERROUND(state[3], state[4], state[9], state[14]) + } + + for (i = 0; i < 16; i++) + x->input[i] = PLUS(x->input[i], state[i]); + + for (i = 0; i < len; i += 4) { + _private_tls_U32TO8_LITTLE(c + i, x->input[i/4]); + } +} + +static inline int poly1305_generate_key(unsigned char *key256, unsigned char *nonce, unsigned int noncelen, unsigned char *poly_key, unsigned int counter) { + struct chacha_ctx ctx; + uint64_t ctr; + memset(&ctx, 0, sizeof(ctx)); + chacha_keysetup(&ctx, key256, 256); + switch (noncelen) { + case 8: + ctr = counter; + chacha_ivsetup(&ctx, nonce, (unsigned char *)&ctr); + break; + case 12: + chacha_ivsetup_96bitnonce(&ctx, nonce, (unsigned char *)&counter); + break; + default: + return -1; + } + chacha20_block(&ctx, poly_key, POLY1305_KEYLEN); + return 0; +} + +/* 17 + sizeof(size_t) + 14*sizeof(unsigned long) */ +typedef struct poly1305_state_internal_t { + unsigned long r[5]; + unsigned long h[5]; + unsigned long pad[4]; + size_t leftover; + unsigned char buffer[poly1305_block_size]; + unsigned char final; +} poly1305_state_internal_t; + +/* interpret four 8 bit unsigned integers as a 32 bit unsigned integer in little endian */ +static unsigned long _private_tls_U8TO32(const unsigned char *p) { + return + (((unsigned long)(p[0] & 0xff) ) | + ((unsigned long)(p[1] & 0xff) << 8) | + ((unsigned long)(p[2] & 0xff) << 16) | + ((unsigned long)(p[3] & 0xff) << 24)); +} + +/* store a 32 bit unsigned integer as four 8 bit unsigned integers in little endian */ +static void _private_tls_U32TO8(unsigned char *p, unsigned long v) { + p[0] = (v ) & 0xff; + p[1] = (v >> 8) & 0xff; + p[2] = (v >> 16) & 0xff; + p[3] = (v >> 24) & 0xff; +} + +void _private_tls_poly1305_init(poly1305_context *ctx, const unsigned char key[32]) { + poly1305_state_internal_t *st = (poly1305_state_internal_t *)ctx; + + /* r &= 0xffffffc0ffffffc0ffffffc0fffffff */ + st->r[0] = (_private_tls_U8TO32(&key[ 0]) ) & 0x3ffffff; + st->r[1] = (_private_tls_U8TO32(&key[ 3]) >> 2) & 0x3ffff03; + st->r[2] = (_private_tls_U8TO32(&key[ 6]) >> 4) & 0x3ffc0ff; + st->r[3] = (_private_tls_U8TO32(&key[ 9]) >> 6) & 0x3f03fff; + st->r[4] = (_private_tls_U8TO32(&key[12]) >> 8) & 0x00fffff; + + /* h = 0 */ + st->h[0] = 0; + st->h[1] = 0; + st->h[2] = 0; + st->h[3] = 0; + st->h[4] = 0; + + /* save pad for later */ + st->pad[0] = _private_tls_U8TO32(&key[16]); + st->pad[1] = _private_tls_U8TO32(&key[20]); + st->pad[2] = _private_tls_U8TO32(&key[24]); + st->pad[3] = _private_tls_U8TO32(&key[28]); + + st->leftover = 0; + st->final = 0; +} + +static void _private_tls_poly1305_blocks(poly1305_state_internal_t *st, const unsigned char *m, size_t bytes) { + const unsigned long hibit = (st->final) ? 0 : (1UL << 24); /* 1 << 128 */ + unsigned long r0,r1,r2,r3,r4; + unsigned long s1,s2,s3,s4; + unsigned long h0,h1,h2,h3,h4; + unsigned long long d0,d1,d2,d3,d4; + unsigned long c; + + r0 = st->r[0]; + r1 = st->r[1]; + r2 = st->r[2]; + r3 = st->r[3]; + r4 = st->r[4]; + + s1 = r1 * 5; + s2 = r2 * 5; + s3 = r3 * 5; + s4 = r4 * 5; + + h0 = st->h[0]; + h1 = st->h[1]; + h2 = st->h[2]; + h3 = st->h[3]; + h4 = st->h[4]; + + while (bytes >= poly1305_block_size) { + /* h += m[i] */ + h0 += (_private_tls_U8TO32(m+ 0) ) & 0x3ffffff; + h1 += (_private_tls_U8TO32(m+ 3) >> 2) & 0x3ffffff; + h2 += (_private_tls_U8TO32(m+ 6) >> 4) & 0x3ffffff; + h3 += (_private_tls_U8TO32(m+ 9) >> 6) & 0x3ffffff; + h4 += (_private_tls_U8TO32(m+12) >> 8) | hibit; + + /* h *= r */ + d0 = ((unsigned long long)h0 * r0) + ((unsigned long long)h1 * s4) + ((unsigned long long)h2 * s3) + ((unsigned long long)h3 * s2) + ((unsigned long long)h4 * s1); + d1 = ((unsigned long long)h0 * r1) + ((unsigned long long)h1 * r0) + ((unsigned long long)h2 * s4) + ((unsigned long long)h3 * s3) + ((unsigned long long)h4 * s2); + d2 = ((unsigned long long)h0 * r2) + ((unsigned long long)h1 * r1) + ((unsigned long long)h2 * r0) + ((unsigned long long)h3 * s4) + ((unsigned long long)h4 * s3); + d3 = ((unsigned long long)h0 * r3) + ((unsigned long long)h1 * r2) + ((unsigned long long)h2 * r1) + ((unsigned long long)h3 * r0) + ((unsigned long long)h4 * s4); + d4 = ((unsigned long long)h0 * r4) + ((unsigned long long)h1 * r3) + ((unsigned long long)h2 * r2) + ((unsigned long long)h3 * r1) + ((unsigned long long)h4 * r0); + + /* (partial) h %= p */ + c = (unsigned long)(d0 >> 26); h0 = (unsigned long)d0 & 0x3ffffff; + d1 += c; c = (unsigned long)(d1 >> 26); h1 = (unsigned long)d1 & 0x3ffffff; + d2 += c; c = (unsigned long)(d2 >> 26); h2 = (unsigned long)d2 & 0x3ffffff; + d3 += c; c = (unsigned long)(d3 >> 26); h3 = (unsigned long)d3 & 0x3ffffff; + d4 += c; c = (unsigned long)(d4 >> 26); h4 = (unsigned long)d4 & 0x3ffffff; + h0 += c * 5; c = (h0 >> 26); h0 = h0 & 0x3ffffff; + h1 += c; + + m += poly1305_block_size; + bytes -= poly1305_block_size; + } + + st->h[0] = h0; + st->h[1] = h1; + st->h[2] = h2; + st->h[3] = h3; + st->h[4] = h4; +} + +void _private_tls_poly1305_finish(poly1305_context *ctx, unsigned char mac[16]) { + poly1305_state_internal_t *st = (poly1305_state_internal_t *)ctx; + unsigned long h0,h1,h2,h3,h4,c; + unsigned long g0,g1,g2,g3,g4; + unsigned long long f; + unsigned long mask; + + /* process the remaining block */ + if (st->leftover) { + size_t i = st->leftover; + st->buffer[i++] = 1; + for (; i < poly1305_block_size; i++) + st->buffer[i] = 0; + st->final = 1; + _private_tls_poly1305_blocks(st, st->buffer, poly1305_block_size); + } + + /* fully carry h */ + h0 = st->h[0]; + h1 = st->h[1]; + h2 = st->h[2]; + h3 = st->h[3]; + h4 = st->h[4]; + + c = h1 >> 26; h1 = h1 & 0x3ffffff; + h2 += c; c = h2 >> 26; h2 = h2 & 0x3ffffff; + h3 += c; c = h3 >> 26; h3 = h3 & 0x3ffffff; + h4 += c; c = h4 >> 26; h4 = h4 & 0x3ffffff; + h0 += c * 5; c = h0 >> 26; h0 = h0 & 0x3ffffff; + h1 += c; + + /* compute h + -p */ + g0 = h0 + 5; c = g0 >> 26; g0 &= 0x3ffffff; + g1 = h1 + c; c = g1 >> 26; g1 &= 0x3ffffff; + g2 = h2 + c; c = g2 >> 26; g2 &= 0x3ffffff; + g3 = h3 + c; c = g3 >> 26; g3 &= 0x3ffffff; + g4 = h4 + c - (1UL << 26); + + /* select h if h < p, or h + -p if h >= p */ + mask = (g4 >> ((sizeof(unsigned long) * 8) - 1)) - 1; + g0 &= mask; + g1 &= mask; + g2 &= mask; + g3 &= mask; + g4 &= mask; + mask = ~mask; + h0 = (h0 & mask) | g0; + h1 = (h1 & mask) | g1; + h2 = (h2 & mask) | g2; + h3 = (h3 & mask) | g3; + h4 = (h4 & mask) | g4; + + /* h = h % (2^128) */ + h0 = ((h0 ) | (h1 << 26)) & 0xffffffff; + h1 = ((h1 >> 6) | (h2 << 20)) & 0xffffffff; + h2 = ((h2 >> 12) | (h3 << 14)) & 0xffffffff; + h3 = ((h3 >> 18) | (h4 << 8)) & 0xffffffff; + + /* mac = (h + pad) % (2^128) */ + f = (unsigned long long)h0 + st->pad[0] ; h0 = (unsigned long)f; + f = (unsigned long long)h1 + st->pad[1] + (f >> 32); h1 = (unsigned long)f; + f = (unsigned long long)h2 + st->pad[2] + (f >> 32); h2 = (unsigned long)f; + f = (unsigned long long)h3 + st->pad[3] + (f >> 32); h3 = (unsigned long)f; + + _private_tls_U32TO8(mac + 0, h0); + _private_tls_U32TO8(mac + 4, h1); + _private_tls_U32TO8(mac + 8, h2); + _private_tls_U32TO8(mac + 12, h3); + + /* zero out the state */ + st->h[0] = 0; + st->h[1] = 0; + st->h[2] = 0; + st->h[3] = 0; + st->h[4] = 0; + st->r[0] = 0; + st->r[1] = 0; + st->r[2] = 0; + st->r[3] = 0; + st->r[4] = 0; + st->pad[0] = 0; + st->pad[1] = 0; + st->pad[2] = 0; + st->pad[3] = 0; +} + +void _private_tls_poly1305_update(poly1305_context *ctx, const unsigned char *m, size_t bytes) { + poly1305_state_internal_t *st = (poly1305_state_internal_t *)ctx; + size_t i; + /* handle leftover */ + if (st->leftover) { + size_t want = (poly1305_block_size - st->leftover); + if (want > bytes) + want = bytes; + for (i = 0; i < want; i++) + st->buffer[st->leftover + i] = m[i]; + bytes -= want; + m += want; + st->leftover += want; + if (st->leftover < poly1305_block_size) + return; + _private_tls_poly1305_blocks(st, st->buffer, poly1305_block_size); + st->leftover = 0; + } + + /* process full blocks */ + if (bytes >= poly1305_block_size) { + size_t want = (bytes & ~(poly1305_block_size - 1)); + _private_tls_poly1305_blocks(st, m, want); + m += want; + bytes -= want; + } + + /* store leftover */ + if (bytes) { + for (i = 0; i < bytes; i++) + st->buffer[st->leftover + i] = m[i]; + st->leftover += bytes; + } +} + +int poly1305_verify(const unsigned char mac1[16], const unsigned char mac2[16]) { + size_t i; + unsigned int dif = 0; + for (i = 0; i < 16; i++) + dif |= (mac1[i] ^ mac2[i]); + dif = (dif - 1) >> ((sizeof(unsigned int) * 8) - 1); + return (dif & 1); +} + +void chacha20_poly1305_key(struct chacha_ctx *ctx, unsigned char *poly1305_key) { + unsigned char key[32]; + unsigned char nonce[12]; + chacha_key(ctx, key); + chacha_nonce(ctx, nonce); + poly1305_generate_key(key, nonce, sizeof(nonce), poly1305_key, 0); +} + +int chacha20_poly1305_aead(struct chacha_ctx *ctx, unsigned char *pt, unsigned int len, unsigned char *aad, unsigned int aad_len, unsigned char *poly_key, unsigned char *out) { + static unsigned char zeropad[] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; + if (aad_len > POLY1305_MAX_AAD) + return -1; + + unsigned int counter = 1; + chacha_ivsetup_96bitnonce(ctx, NULL, (unsigned char *)&counter); + chacha_encrypt_bytes(ctx, pt, out, len); + + poly1305_context aead_ctx; + _private_tls_poly1305_init(&aead_ctx, poly_key); + _private_tls_poly1305_update(&aead_ctx, aad, aad_len); + int rem = aad_len % 16; + if (rem) + _private_tls_poly1305_update(&aead_ctx, zeropad, 16 - rem); + _private_tls_poly1305_update(&aead_ctx, out, len); + rem = len % 16; + if (rem) + _private_tls_poly1305_update(&aead_ctx, zeropad, 16 - rem); + + unsigned char trail[16]; + _private_tls_U32TO8(trail, aad_len); + *(int *)(trail + 4) = 0; + _private_tls_U32TO8(trail + 8, len); + *(int *)(trail + 12) = 0; + + _private_tls_poly1305_update(&aead_ctx, trail, 16); + _private_tls_poly1305_finish(&aead_ctx, out + len); + + return len + POLY1305_TAGLEN; +} +#endif +#endif + +typedef enum { + KEA_dhe_dss, + KEA_dhe_rsa, + KEA_dh_anon, + KEA_rsa, + KEA_dh_dss, + KEA_dh_rsa, + KEA_ec_diffie_hellman +} KeyExchangeAlgorithm; + +typedef enum { + rsa_sign = 1, + dss_sign = 2, + rsa_fixed_dh = 3, + dss_fixed_dh = 4, + rsa_ephemeral_dh_RESERVED = 5, + dss_ephemeral_dh_RESERVED = 6, + fortezza_dms_RESERVED = 20, + ecdsa_sign = 64, + rsa_fixed_ecdh = 65, + ecdsa_fixed_ecdh = 66 +} TLSClientCertificateType; + +typedef enum { + none = 0, + md5 = 1, + sha1 = 2, + sha224 = 3, + sha256 = 4, + sha384 = 5, + sha512 = 6, + _md5_sha1 = 255 +} TLSHashAlgorithm; + +#define TLS_HASH_ALGO_NUMBER (sha512 - md5 + 1) + +typedef enum { + anonymous = 0, + rsa = 1, + dsa = 2, + ecdsa = 3 +} TLSSignatureAlgorithm; + +#define TLS_SIGN_ALGO_NUMBER (ecdsa - rsa + 1) + +struct _private_OID_chain { + void *top; + unsigned char *oid; +}; + +struct TLSCertificate { + unsigned short version; + unsigned int algorithm; + unsigned int key_algorithm; + unsigned int ec_algorithm; + unsigned char *exponent; + unsigned int exponent_len; + unsigned char *pk; + unsigned int pk_len; + unsigned char *priv; + unsigned int priv_len; + unsigned char *issuer_country; + unsigned char *issuer_state; + unsigned char *issuer_location; + unsigned char *issuer_entity; + unsigned char *issuer_subject; + unsigned char *not_before; + unsigned char *not_after; + unsigned char *country; + unsigned char *state; + unsigned char *location; + unsigned char *entity; + unsigned char *subject; + unsigned char **san; + unsigned short san_length; + unsigned char *ocsp; + unsigned char *serial_number; + unsigned int serial_len; + unsigned char *sign_key; + unsigned int sign_len; + unsigned char *fingerprint; + unsigned char *der_bytes; + unsigned int der_len; + unsigned char *bytes; + unsigned int len; +}; + +typedef struct { + union { + symmetric_CBC aes_local; + gcm_state aes_gcm_local; +#ifdef TLS_WITH_CHACHA20_POLY1305 + chacha_ctx chacha_local; +#endif + } ctx_local; + union { + symmetric_CBC aes_remote; + gcm_state aes_gcm_remote; +#ifdef TLS_WITH_CHACHA20_POLY1305 + chacha_ctx chacha_remote; +#endif + } ctx_remote; + union { + unsigned char local_mac[TLS_MAX_MAC_SIZE]; + unsigned char local_aead_iv[TLS_AES_GCM_IV_LENGTH]; +#ifdef WITH_TLS_13 + unsigned char local_iv[TLS_13_AES_GCM_IV_LENGTH]; +#endif +#ifdef TLS_WITH_CHACHA20_POLY1305 + unsigned char local_nonce[TLS_CHACHA20_IV_LENGTH]; +#endif + } ctx_local_mac; + union { + unsigned char remote_aead_iv[TLS_AES_GCM_IV_LENGTH]; + unsigned char remote_mac[TLS_MAX_MAC_SIZE]; +#ifdef WITH_TLS_13 + unsigned char remote_iv[TLS_13_AES_GCM_IV_LENGTH]; +#endif +#ifdef TLS_WITH_CHACHA20_POLY1305 + unsigned char remote_nonce[TLS_CHACHA20_IV_LENGTH]; +#endif + } ctx_remote_mac; + unsigned char created; +} TLSCipher; + +typedef struct { + hash_state hash32; + hash_state hash48; +#ifdef TLS_LEGACY_SUPPORT + hash_state hash2; +#endif + unsigned char created; +} TLSHash; + +#ifdef TLS_FORWARD_SECRECY +#define mp_init(a) ltc_mp.init(a) +#define mp_init_multi ltc_init_multi +#define mp_clear(a) ltc_mp.deinit(a) +#define mp_clear_multi ltc_deinit_multi +#define mp_count_bits(a) ltc_mp.count_bits(a) +#define mp_read_radix(a, b, c) ltc_mp.read_radix(a, b, c) +#define mp_unsigned_bin_size(a) ltc_mp.unsigned_size(a) +#define mp_to_unsigned_bin(a, b) ltc_mp.unsigned_write(a, b) +#define mp_read_unsigned_bin(a, b, c) ltc_mp.unsigned_read(a, b, c) +#define mp_exptmod(a, b, c, d) ltc_mp.exptmod(a, b, c, d) +#define mp_add(a, b, c) ltc_mp.add(a, b, c) +#define mp_mul(a, b, c) ltc_mp.mul(a, b, c) +#define mp_cmp(a, b) ltc_mp.compare(a, b) +#define mp_cmp_d(a, b) ltc_mp.compare_d(a, b) +#define mp_sqr(a, b) ltc_mp.sqr(a, b) +#define mp_mod(a, b, c) ltc_mp.mpdiv(a, b, NULL, c) +#define mp_sub(a, b, c) ltc_mp.sub(a, b, c) +#define mp_set(a, b) ltc_mp.set_int(a, b) +#define mp_copy(a, b) ltc_mp.copy(a, b) +#define mp_submod(a, b, c, d) ltc_mp.submod(a, b, c, d) +#define mp_mulmod(a, b, c, d) ltc_mp.mulmod(a, b, c, d) +#define mp_addmod(a, b, c, d) ltc_mp.addmod(a, b, c, d) + +typedef struct { + int iana; + void *x; + void *y; + void *p; + void *g; +} DHKey; + +#ifdef WITH_TLS_13 +static DHKey ffdhe2048 = { + 0x0100, + NULL, + NULL, + (void *)"FFFFFFFFFFFFFFFFADF85458A2BB4A9AAFDC5620273D3CF1D8B9C583CE2D3695A9E13641146433FBCC939DCE249B3EF97D2FE363630C75D8F681B202AEC4617AD3DF1ED5D5FD65612433F51F5F066ED0856365553DED1AF3B557135E7F57C935984F0C70E0E68B77E2A689DAF3EFE8721DF158A136ADE73530ACCA4F483A797ABC0AB182B324FB61D108A94BB2C8E3FBB96ADAB760D7F4681D4F42A3DE394DF4AE56EDE76372BB190B07A7C8EE0A6D709E02FCE1CDF7E2ECC03404CD28342F619172FE9CE98583FF8E4F1232EEF28183C3FE3B1B4C6FAD733BB5FCBC2EC22005C58EF1837D1683B2C6F34A26C1B2EFFA886B423861285C97FFFFFFFFFFFFFFFF", + (void *)"00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002" +}; + +static DHKey ffdhe3072 = { + 0x0101, + NULL, + NULL, + (void *)"FFFFFFFFFFFFFFFFADF85458A2BB4A9AAFDC5620273D3CF1D8B9C583CE2D3695A9E13641146433FBCC939DCE249B3EF97D2FE363630C75D8F681B202AEC4617AD3DF1ED5D5FD65612433F51F5F066ED0856365553DED1AF3B557135E7F57C935984F0C70E0E68B77E2A689DAF3EFE8721DF158A136ADE73530ACCA4F483A797ABC0AB182B324FB61D108A94BB2C8E3FBB96ADAB760D7F4681D4F42A3DE394DF4AE56EDE76372BB190B07A7C8EE0A6D709E02FCE1CDF7E2ECC03404CD28342F619172FE9CE98583FF8E4F1232EEF28183C3FE3B1B4C6FAD733BB5FCBC2EC22005C58EF1837D1683B2C6F34A26C1B2EFFA886B4238611FCFDCDE355B3B6519035BBC34F4DEF99C023861B46FC9D6E6C9077AD91D2691F7F7EE598CB0FAC186D91CAEFE130985139270B4130C93BC437944F4FD4452E2D74DD364F2E21E71F54BFF5CAE82AB9C9DF69EE86D2BC522363A0DABC521979B0DEADA1DBF9A42D5C4484E0ABCD06BFA53DDEF3C1B20EE3FD59D7C25E41D2B66C62E37FFFFFFFFFFFFFFFF", + (void *)"000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002" +}; + +static DHKey ffdhe4096 = { + 0x0102, + NULL, + NULL, + (void *)"FFFFFFFFFFFFFFFFADF85458A2BB4A9AAFDC5620273D3CF1D8B9C583CE2D3695A9E13641146433FBCC939DCE249B3EF97D2FE363630C75D8F681B202AEC4617AD3DF1ED5D5FD65612433F51F5F066ED0856365553DED1AF3B557135E7F57C935984F0C70E0E68B77E2A689DAF3EFE8721DF158A136ADE73530ACCA4F483A797ABC0AB182B324FB61D108A94BB2C8E3FBB96ADAB760D7F4681D4F42A3DE394DF4AE56EDE76372BB190B07A7C8EE0A6D709E02FCE1CDF7E2ECC03404CD28342F619172FE9CE98583FF8E4F1232EEF28183C3FE3B1B4C6FAD733BB5FCBC2EC22005C58EF1837D1683B2C6F34A26C1B2EFFA886B4238611FCFDCDE355B3B6519035BBC34F4DEF99C023861B46FC9D6E6C9077AD91D2691F7F7EE598CB0FAC186D91CAEFE130985139270B4130C93BC437944F4FD4452E2D74DD364F2E21E71F54BFF5CAE82AB9C9DF69EE86D2BC522363A0DABC521979B0DEADA1DBF9A42D5C4484E0ABCD06BFA53DDEF3C1B20EE3FD59D7C25E41D2B669E1EF16E6F52C3164DF4FB7930E9E4E58857B6AC7D5F42D69F6D187763CF1D5503400487F55BA57E31CC7A7135C886EFB4318AED6A1E012D9E6832A907600A918130C46DC778F971AD0038092999A333CB8B7A1A1DB93D7140003C2A4ECEA9F98D0ACC0A8291CDCEC97DCF8EC9B55A7F88A46B4DB5A851F44182E1C68A007E5E655F6AFFFFFFFFFFFFFFFF", + (void *)"0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002" +}; + +static DHKey ffdhe6144 = { + 0x0103, + NULL, + NULL, + (void *)"FFFFFFFFFFFFFFFFADF85458A2BB4A9AAFDC5620273D3CF1D8B9C583CE2D3695A9E13641146433FBCC939DCE249B3EF97D2FE363630C75D8F681B202AEC4617AD3DF1ED5D5FD65612433F51F5F066ED0856365553DED1AF3B557135E7F57C935984F0C70E0E68B77E2A689DAF3EFE8721DF158A136ADE73530ACCA4F483A797ABC0AB182B324FB61D108A94BB2C8E3FBB96ADAB760D7F4681D4F42A3DE394DF4AE56EDE76372BB190B07A7C8EE0A6D709E02FCE1CDF7E2ECC03404CD28342F619172FE9CE98583FF8E4F1232EEF28183C3FE3B1B4C6FAD733BB5FCBC2EC22005C58EF1837D1683B2C6F34A26C1B2EFFA886B4238611FCFDCDE355B3B6519035BBC34F4DEF99C023861B46FC9D6E6C9077AD91D2691F7F7EE598CB0FAC186D91CAEFE130985139270B4130C93BC437944F4FD4452E2D74DD364F2E21E71F54BFF5CAE82AB9C9DF69EE86D2BC522363A0DABC521979B0DEADA1DBF9A42D5C4484E0ABCD06BFA53DDEF3C1B20EE3FD59D7C25E41D2B669E1EF16E6F52C3164DF4FB7930E9E4E58857B6AC7D5F42D69F6D187763CF1D5503400487F55BA57E31CC7A7135C886EFB4318AED6A1E012D9E6832A907600A918130C46DC778F971AD0038092999A333CB8B7A1A1DB93D7140003C2A4ECEA9F98D0ACC0A8291CDCEC97DCF8EC9B55A7F88A46B4DB5A851F44182E1C68A007E5E0DD9020BFD64B645036C7A4E677D2C38532A3A23BA4442CAF53EA63BB454329B7624C8917BDD64B1C0FD4CB38E8C334C701C3ACDAD0657FCCFEC719B1F5C3E4E46041F388147FB4CFDB477A52471F7A9A96910B855322EDB6340D8A00EF092350511E30ABEC1FFF9E3A26E7FB29F8C183023C3587E38DA0077D9B4763E4E4B94B2BBC194C6651E77CAF992EEAAC0232A281BF6B3A739C1226116820AE8DB5847A67CBEF9C9091B462D538CD72B03746AE77F5E62292C311562A846505DC82DB854338AE49F5235C95B91178CCF2DD5CACEF403EC9D1810C6272B045B3B71F9DC6B80D63FDD4A8E9ADB1E6962A69526D43161C1A41D570D7938DAD4A40E329CD0E40E65FFFFFFFFFFFFFFFF", + (void *)"000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002" +}; + +static DHKey ffdhe8192 = { + 0x0104, + NULL, + NULL, + (void *)"FFFFFFFFFFFFFFFFADF85458A2BB4A9AAFDC5620273D3CF1D8B9C583CE2D3695A9E13641146433FBCC939DCE249B3EF97D2FE363630C75D8F681B202AEC4617AD3DF1ED5D5FD65612433F51F5F066ED0856365553DED1AF3B557135E7F57C935984F0C70E0E68B77E2A689DAF3EFE8721DF158A136ADE73530ACCA4F483A797ABC0AB182B324FB61D108A94BB2C8E3FBB96ADAB760D7F4681D4F42A3DE394DF4AE56EDE76372BB190B07A7C8EE0A6D709E02FCE1CDF7E2ECC03404CD28342F619172FE9CE98583FF8E4F1232EEF28183C3FE3B1B4C6FAD733BB5FCBC2EC22005C58EF1837D1683B2C6F34A26C1B2EFFA886B4238611FCFDCDE355B3B6519035BBC34F4DEF99C023861B46FC9D6E6C9077AD91D2691F7F7EE598CB0FAC186D91CAEFE130985139270B4130C93BC437944F4FD4452E2D74DD364F2E21E71F54BFF5CAE82AB9C9DF69EE86D2BC522363A0DABC521979B0DEADA1DBF9A42D5C4484E0ABCD06BFA53DDEF3C1B20EE3FD59D7C25E41D2B669E1EF16E6F52C3164DF4FB7930E9E4E58857B6AC7D5F42D69F6D187763CF1D5503400487F55BA57E31CC7A7135C886EFB4318AED6A1E012D9E6832A907600A918130C46DC778F971AD0038092999A333CB8B7A1A1DB93D7140003C2A4ECEA9F98D0ACC0A8291CDCEC97DCF8EC9B55A7F88A46B4DB5A851F44182E1C68A007E5E0DD9020BFD64B645036C7A4E677D2C38532A3A23BA4442CAF53EA63BB454329B7624C8917BDD64B1C0FD4CB38E8C334C701C3ACDAD0657FCCFEC719B1F5C3E4E46041F388147FB4CFDB477A52471F7A9A96910B855322EDB6340D8A00EF092350511E30ABEC1FFF9E3A26E7FB29F8C183023C3587E38DA0077D9B4763E4E4B94B2BBC194C6651E77CAF992EEAAC0232A281BF6B3A739C1226116820AE8DB5847A67CBEF9C9091B462D538CD72B03746AE77F5E62292C311562A846505DC82DB854338AE49F5235C95B91178CCF2DD5CACEF403EC9D1810C6272B045B3B71F9DC6B80D63FDD4A8E9ADB1E6962A69526D43161C1A41D570D7938DAD4A40E329CCFF46AAA36AD004CF600C8381E425A31D951AE64FDB23FCEC9509D43687FEB69EDD1CC5E0B8CC3BDF64B10EF86B63142A3AB8829555B2F747C932665CB2C0F1CC01BD70229388839D2AF05E454504AC78B7582822846C0BA35C35F5C59160CC046FD8251541FC68C9C86B022BB7099876A460E7451A8A93109703FEE1C217E6C3826E52C51AA691E0E423CFC99E9E31650C1217B624816CDAD9A95F9D5B8019488D9C0A0A1FE3075A577E23183F81D4A3F2FA4571EFC8CE0BA8A4FE8B6855DFE72B0A66EDED2FBABFBE58A30FAFABE1C5D71A87E2F741EF8C1FE86FEA6BBFDE530677F0D97D11D49F7A8443D0822E506A9F4614E011E2A94838FF88CD68C8BB7C5C6424CFFFFFFFFFFFFFFFF", + (void *)"00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002" +}; +#endif + +#if CRYPT > 0x0117 + #define ltc_ecc_set_type ltc_ecc_curve +#endif + +struct ECCCurveParameters { + int size; + int iana; + const char *name; + const char *P; + const char *A; + const char *B; + const char *Gx; + const char *Gy; + const char *order; + const char *oid; + ltc_ecc_set_type dp; +}; + +static struct ECCCurveParameters secp192r1 = { + 24, + 19, + "secp192r1", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFF", // P + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFC", // A + "64210519E59C80E70FA7E9AB72243049FEB8DEECC146B9B1", // B + "188DA80EB03090F67CBF20EB43A18800F4FF0AFD82FF1012", // Gx + "07192B95FFC8DA78631011ED6B24CDD573F977A11E794811", // Gy + "FFFFFFFFFFFFFFFFFFFFFFFF99DEF836146BC9B1B4D22831", // order (n) + "1.2.840.10045.3.1.1" // oid +}; + + +static struct ECCCurveParameters secp224r1 = { + 28, + 21, + "secp224r1", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF000000000000000000000001", // P + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFFFFFFFFFE", // A + "B4050A850C04B3ABF54132565044B0B7D7BFD8BA270B39432355FFB4", // B + "B70E0CBD6BB4BF7F321390B94A03C1D356C21122343280D6115C1D21", // Gx + "BD376388B5F723FB4C22DFE6CD4375A05A07476444D5819985007E34", // Gy + "FFFFFFFFFFFFFFFFFFFFFFFFFFFF16A2E0B8F03E13DD29455C5C2A3D", // order (n) + "1.3.132.0.33" // oid +}; + +static struct ECCCurveParameters secp224k1 = { + 28, + 20, + "secp224k1", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFE56D", // P + "00000000000000000000000000000000000000000000000000000000", // A + "00000000000000000000000000000000000000000000000000000005", // B + "A1455B334DF099DF30FC28A169A467E9E47075A90F7E650EB6B7A45C", // Gx + "7E089FED7FBA344282CAFBD6F7E319F7C0B0BD59E2CA4BDB556D61A5", // Gy + "0000000000000000000000000001DCE8D2EC6184CAF0A971769FB1F7", // order (n) + "1.3.132.0.32" // oid +}; + +static struct ECCCurveParameters secp256r1 = { + 32, + 23, + "secp256r1", + "FFFFFFFF00000001000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFF", // P + "FFFFFFFF00000001000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFC", // A + "5AC635D8AA3A93E7B3EBBD55769886BC651D06B0CC53B0F63BCE3C3E27D2604B", // B + "6B17D1F2E12C4247F8BCE6E563A440F277037D812DEB33A0F4A13945D898C296", // Gx + "4FE342E2FE1A7F9B8EE7EB4A7C0F9E162BCE33576B315ECECBB6406837BF51F5", // Gy + "FFFFFFFF00000000FFFFFFFFFFFFFFFFBCE6FAADA7179E84F3B9CAC2FC632551", // order (n) + "1.2.840.10045.3.1.7" // oid +}; + +static struct ECCCurveParameters secp256k1 = { + 32, + 22, + "secp256k1", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F", // P + "0000000000000000000000000000000000000000000000000000000000000000", // A + "0000000000000000000000000000000000000000000000000000000000000007", // B + "79BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F81798", // Gx + "483ADA7726A3C4655DA4FBFC0E1108A8FD17B448A68554199C47D08FFB10D4B8", // Gy + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141", // order (n) + "1.3.132.0.10" // oid +}; + +static struct ECCCurveParameters secp384r1 = { + 48, + 24, + "secp384r1", + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFF0000000000000000FFFFFFFF", // P + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFF0000000000000000FFFFFFFC", // A + "B3312FA7E23EE7E4988E056BE3F82D19181D9C6EFE8141120314088F5013875AC656398D8A2ED19D2A85C8EDD3EC2AEF", // B + "AA87CA22BE8B05378EB1C71EF320AD746E1D3B628BA79B9859F741E082542A385502F25DBF55296C3A545E3872760AB7", // Gx + "3617DE4A96262C6F5D9E98BF9292DC29F8F41DBD289A147CE9DA3113B5F0B8C00A60B1CE1D7E819D7A431D7C90EA0E5F", // Gy + "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFC7634D81F4372DDF581A0DB248B0A77AECEC196ACCC52973", // order (n) + "1.3.132.0.34" // oid +}; + +static struct ECCCurveParameters secp521r1 = { + 66, + 25, + "secp521r1", + "01FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF", // P + "01FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFC", // A + "0051953EB9618E1C9A1F929A21A0B68540EEA2DA725B99B315F3B8B489918EF109E156193951EC7E937B1652C0BD3BB1BF073573DF883D2C34F1EF451FD46B503F00", // B + "00C6858E06B70404E9CD9E3ECB662395B4429C648139053FB521F828AF606B4D3DBAA14B5E77EFE75928FE1DC127A2FFA8DE3348B3C1856A429BF97E7E31C2E5BD66", // Gx + "011839296A789A3BC0045C8A5FB42C7D1BD998F54449579B446817AFBD17273E662C97EE72995EF42640C550B9013FAD0761353C7086A272C24088BE94769FD16650", // Gy + "01FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFA51868783BF2F966B7FCC0148F709A5D03BB5C9B8899C47AEBB6FB71E91386409", // order (n) + "1.3.132.0.35" // oid +}; + +#ifdef TLS_CURVE25519 +// dummy +static struct ECCCurveParameters x25519 = { + 32, + 29, + "x25519", + "7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFED", // P + "0000000000000000000000000000000000000000000000000000000000076D06", // A + "0000000000000000000000000000000000000000000000000000000000000000", // B + "0000000000000000000000000000000000000000000000000000000000000009", // Gx + "20AE19A1B8A086B4E01EDD2C7748D14C923D4D7E6D7C61B229E9C5A27ECED3D9", // Gy + "1000000000000000000000000000000014DEF9DEA2F79CD65812631A5CF5D3ED", // order (n) + "1.3.101.110" // oid +}; +#endif + +static struct ECCCurveParameters * const default_curve = &secp256r1; + +void init_curve(struct ECCCurveParameters *curve) { +#if CRYPT < 0x0118 + curve->dp.size = curve->size; + curve->dp.name = (char *)curve->name; +#else + curve->dp.cofactor = 1; + curve->dp.A = (char *)curve->A; + curve->dp.OID = curve->oid; +#endif + curve->dp.B = (char *)curve->B; + curve->dp.prime = (char *)curve->P; + curve->dp.Gx = (char *)curve->Gx; + curve->dp.Gy = (char *)curve->Gy; + curve->dp.order = (char *)curve->order; +} + +void init_curves() { + init_curve(&secp192r1); + init_curve(&secp224r1); + init_curve(&secp224k1); + init_curve(&secp256r1); + init_curve(&secp256k1); + init_curve(&secp384r1); + init_curve(&secp521r1); +} +#endif + +struct DTLSFragment { + char *buffer; + int len; + int written; +}; + +struct TLSHandshakeList { + unsigned char connection_status; + unsigned char direction; + unsigned char *msg; + unsigned int len; + void *next; +}; + +struct DTLSData { + struct TLSHandshakeList *dtls_handshake_list; + struct DTLSFragment *fragment; + unsigned char *key_exchange; + unsigned int key_exchange_len; +#ifdef TLS_DTLS_EXTENDED_MASTER_SECRET + unsigned char extended_master_secret; +#endif + unsigned char has_random; + char *remote_fingerprint; +}; + +struct TLSContext { + unsigned char remote_random[TLS_CLIENT_RANDOM_SIZE]; + unsigned char local_random[TLS_SERVER_RANDOM_SIZE]; + unsigned char session[TLS_MAX_SESSION_ID]; + unsigned char session_size; + unsigned short cipher; + unsigned short version; + unsigned char is_server; + struct TLSCertificate **certificates; + struct TLSCertificate *private_key; +#ifdef TLS_ECDSA_SUPPORTED + struct TLSCertificate *ec_private_key; +#endif +#ifdef TLS_FORWARD_SECRECY + DHKey *dhe; + ecc_key *ecc_dhe; + char *default_dhe_p; + char *default_dhe_g; + const struct ECCCurveParameters *curve; +#endif + struct TLSCertificate **client_certificates; + unsigned int certificates_count; + unsigned int client_certificates_count; + unsigned char *master_key; + unsigned int master_key_len; + unsigned char *premaster_key; + unsigned int premaster_key_len; + unsigned char cipher_spec_set; + TLSCipher crypto; + TLSHash *handshake_hash; + + unsigned char *message_buffer; + unsigned int message_buffer_len; + uint64_t remote_sequence_number; + uint64_t local_sequence_number; + + unsigned char connection_status; + unsigned char critical_error; + unsigned char error_code; + + unsigned char *tls_buffer; + unsigned int tls_buffer_len; + + unsigned char *application_buffer; + unsigned int application_buffer_len; + unsigned char is_child; + unsigned char exportable; + unsigned char *exportable_keys; + unsigned char exportable_size; + char *sni; + unsigned char request_client_certificate; + unsigned char dtls; + unsigned short dtls_epoch_local; + unsigned short dtls_epoch_remote; + unsigned char *dtls_cookie; + unsigned char dtls_cookie_len; + unsigned char dtls_seq; + unsigned char *cached_handshake; + unsigned int cached_handshake_len; + unsigned char client_verified; + // handshake messages flags + unsigned char hs_messages[11]; + + void *user_data; + struct TLSCertificate **root_certificates; + unsigned int root_count; +#ifdef TLS_ACCEPT_SECURE_RENEGOTIATION + unsigned char *verify_data; + unsigned char verify_len; +#endif +#ifdef WITH_TLS_13 + unsigned char *finished_key; + unsigned char *remote_finished_key; + unsigned char *server_finished_hash; +#endif +#ifdef TLS_CURVE25519 + unsigned char *client_secret; +#endif + char **alpn; + unsigned char alpn_count; + char *negotiated_alpn; + unsigned int sleep_until; + unsigned short tls13_version; +#ifdef TLS_12_FALSE_START + unsigned char false_start; +#endif + + struct DTLSData *dtls_data; +}; + +struct TLSPacket { + unsigned char *buf; + unsigned int len; + unsigned int size; + unsigned char broken; + struct TLSContext *context; +}; + +struct TLSRTCPeerBuffer { + unsigned char *buf; + unsigned int len; + + void *next; +}; + +#define SRTP_MASTER_KEY_KEY_LEN 16 +#define SRTP_MASTER_KEY_SALT_LEN 14 + +struct TLSRTCPeerConnection { + struct TLSContext *context; + unsigned char stun_transcation_id[12]; + + char local_user[5]; + char local_pwd[25]; + + unsigned char *remote_user; + int remote_user_len; + unsigned char *remote_pwd; + int remote_pwd_len; + + tls_validation_function certificate_verify; + + void *userdata; + + unsigned char local_state; + unsigned char remote_state; + + unsigned char active; + +#ifdef TLS_SRTP + struct SRTPContext *srtp_local; + struct SRTPContext *srtp_remote; +#endif + + struct TLSRTCPeerBuffer *write_buffer; + struct TLSRTCPeerBuffer *read_buffer; +}; + +#ifdef SSL_COMPATIBLE_INTERFACE + +typedef int (*SOCKET_RECV_CALLBACK)(int socket, void *buffer, size_t length, int flags); +typedef int (*SOCKET_SEND_CALLBACK)(int socket, const void *buffer, size_t length, int flags); + +#ifndef _WIN32 +#include +#endif +#endif + +static const unsigned int version_id[] = {1, 1, 1, 0}; +static const unsigned int pk_id[] = {1, 1, 7, 0}; +static const unsigned int serial_id[] = {1, 1, 2, 1, 0}; +static const unsigned int issurer_id[] = {1, 1, 4, 0}; +static const unsigned int owner_id[] = {1, 1, 6, 0}; +static const unsigned int validity_id[] = {1, 1, 5, 0}; +static const unsigned int algorithm_id[] = {1, 1, 3, 0}; +static const unsigned int sign_id[] = {1, 3, 2, 1, 0}; +static const unsigned int priv_id[] = {1, 4, 0}; +static const unsigned int priv_der_id[] = {1, 3, 1, 0}; +static const unsigned int ecc_priv_id[] = {1, 2, 0}; + +static const unsigned char country_oid[] = {0x55, 0x04, 0x06, 0x00}; +static const unsigned char state_oid[] = {0x55, 0x04, 0x08, 0x00}; +static const unsigned char location_oid[] = {0x55, 0x04, 0x07, 0x00}; +static const unsigned char entity_oid[] = {0x55, 0x04, 0x0A, 0x00}; +static const unsigned char subject_oid[] = {0x55, 0x04, 0x03, 0x00}; +static const unsigned char san_oid[] = {0x55, 0x1D, 0x11, 0x00}; +static const unsigned char ocsp_oid[] = {0x2B, 0x06, 0x01, 0x05, 0x05, 0x07, 0x30, 0x01, 0x00}; + +static const unsigned char TLS_RSA_SIGN_RSA_OID[] = {0x2A, 0x86, 0x48, 0x86, 0xF7, 0x0D, 0x01, 0x01, 0x01, 0x00}; +static const unsigned char TLS_RSA_SIGN_MD5_OID[] = {0x2A, 0x86, 0x48, 0x86, 0xF7, 0x0D, 0x01, 0x01, 0x04, 0x00}; +static const unsigned char TLS_RSA_SIGN_SHA1_OID[] = {0x2A, 0x86, 0x48, 0x86, 0xF7, 0x0D, 0x01, 0x01, 0x05, 0x00}; +static const unsigned char TLS_RSA_SIGN_SHA256_OID[] = {0x2A, 0x86, 0x48, 0x86, 0xF7, 0x0D, 0x01, 0x01, 0x0B, 0x00}; +static const unsigned char TLS_RSA_SIGN_SHA384_OID[] = {0x2A, 0x86, 0x48, 0x86, 0xF7, 0x0D, 0x01, 0x01, 0x0C, 0x00}; +static const unsigned char TLS_RSA_SIGN_SHA512_OID[] = {0x2A, 0x86, 0x48, 0x86, 0xF7, 0x0D, 0x01, 0x01, 0x0D, 0x00}; + +// static const unsigned char TLS_ECDSA_SIGN_SHA1_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x04, 0x01, 0x05, 0x00, 0x00}; +// static const unsigned char TLS_ECDSA_SIGN_SHA224_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x04, 0x03, 0x01, 0x05, 0x00, 0x00}; +static const unsigned char TLS_ECDSA_SIGN_SHA256_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x04, 0x03, 0x02, 0x05, 0x00, 0x00}; +// static const unsigned char TLS_ECDSA_SIGN_SHA384_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x04, 0x03, 0x03, 0x05, 0x00, 0x00}; +// static const unsigned char TLS_ECDSA_SIGN_SHA512_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x04, 0x03, 0x04, 0x05, 0x00, 0x00}; + +static const unsigned char TLS_EC_PUBLIC_KEY_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x02, 0x01, 0x00}; + +static const unsigned char TLS_EC_prime192v1_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x01, 0x00}; +static const unsigned char TLS_EC_prime192v2_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x02, 0x00}; +static const unsigned char TLS_EC_prime192v3_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x03, 0x00}; +static const unsigned char TLS_EC_prime239v1_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x04, 0x00}; +static const unsigned char TLS_EC_prime239v2_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x05, 0x00}; +static const unsigned char TLS_EC_prime239v3_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x06, 0x00}; +static const unsigned char TLS_EC_prime256v1_OID[] = {0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x07, 0x00}; + +#define TLS_EC_secp256r1_OID TLS_EC_prime256v1_OID +static const unsigned char TLS_EC_secp224r1_OID[] = {0x2B, 0x81, 0x04, 0x00, 0x21, 0x00}; +static const unsigned char TLS_EC_secp384r1_OID[] = {0x2B, 0x81, 0x04, 0x00, 0x22, 0x00}; +static const unsigned char TLS_EC_secp521r1_OID[] = {0x2B, 0x81, 0x04, 0x00, 0x23, 0x00}; + +struct TLSCertificate *asn1_parse(struct TLSContext *context, const unsigned char *buffer, unsigned int size, int client_cert); +int _private_tls_update_hash(struct TLSContext *context, const unsigned char *in, unsigned int len, unsigned char direction, unsigned char connection_status); +struct TLSPacket *tls_build_finished(struct TLSContext *context); +unsigned int _private_tls_hmac_message(unsigned char local, struct TLSContext *context, const unsigned char *buf, int buf_len, const unsigned char *buf2, int buf_len2, unsigned char *out, unsigned int outlen, uint64_t remote_sequence_number); +int tls_random(unsigned char *key, int len); +void tls_destroy_packet(struct TLSPacket *packet); +struct TLSPacket *tls_build_hello(struct TLSContext *context, int tls13_downgrade); +struct TLSPacket *tls_build_certificate(struct TLSContext *context); +struct TLSPacket *tls_build_done(struct TLSContext *context); +struct TLSPacket *tls_build_alert(struct TLSContext *context, char critical, unsigned char code); +struct TLSPacket *tls_build_change_cipher_spec(struct TLSContext *context); +struct TLSPacket *tls_build_verify_request(struct TLSContext *context); +int _private_tls_crypto_create(struct TLSContext *context, int key_length, unsigned char *localkey, unsigned char *localiv, unsigned char *remotekey, unsigned char *remoteiv); +int _private_tls_get_hash(struct TLSContext *context, unsigned char *hout); +int _private_tls_done_hash(struct TLSContext *context, unsigned char *hout); +int _private_tls_get_hash_idx(struct TLSContext *context); +int _private_tls_build_random(struct TLSPacket *packet); +unsigned int _private_tls_mac_length(struct TLSContext *context); +void _private_dtls_handshake_data(struct TLSContext *context, struct TLSPacket *packet, unsigned int dataframe); +#ifdef TLS_FORWARD_SECRECY +void _private_tls_dhe_free(struct TLSContext *context); +void _private_tls_ecc_dhe_free(struct TLSContext *context); +void _private_tls_dh_clear_key(DHKey *key); +#endif + +#ifdef WITH_TLS_13 +struct TLSPacket *tls_build_encrypted_extensions(struct TLSContext *context); +struct TLSPacket *tls_build_certificate_verify(struct TLSContext *context); +#endif + +// dtls base secret +static unsigned char dtls_secret[32]; + +static unsigned char dependecies_loaded = 0; +// not supported +// static unsigned char TLS_DSA_SIGN_SHA1_OID[] = {0x2A, 0x86, 0x52, 0xCE, 0x38, 0x04, 0x03, 0x00}; + +// base64 stuff +static const char cd64[] = "|$$$}rstuvwxyz{$$$$$$$>?@ABCDEFGHIJKLMNOPQRSTUVW$$$$$$XYZ[\\]^_`abcdefghijklmnopq"; + +void _private_b64_decodeblock(unsigned char in[4], unsigned char out[3]) { + out[0] = (unsigned char )(in[0] << 2 | in[1] >> 4); + out[1] = (unsigned char )(in[1] << 4 | in[2] >> 2); + out[2] = (unsigned char )(((in[2] << 6) & 0xc0) | in[3]); +} + +int _private_b64_decode(const char *in_buffer, int in_buffer_size, unsigned char *out_buffer) { + unsigned char in[4], out[3], v; + int i, len; + + const char *ptr = in_buffer; + char *out_ptr = (char *)out_buffer; + + while (ptr <= in_buffer + in_buffer_size) { + for (len = 0, i = 0; i < 4 && (ptr <= in_buffer + in_buffer_size); i++) { + v = 0; + while ((ptr <= in_buffer + in_buffer_size) && v == 0) { + v = (unsigned char)ptr[0]; + ptr++; + v = (unsigned char)((v < 43 || v > 122) ? 0 : cd64[v - 43]); + if (v) + v = (unsigned char)((v == '$') ? 0 : v - 61); + } + if (ptr <= in_buffer + in_buffer_size) { + len++; + if (v) + in[i] = (unsigned char)(v - 1); + } else { + in[i] = 0; + } + } + if (len) { + _private_b64_decodeblock(in, out); + for (i = 0; i < len - 1; i++) { + out_ptr[0] = out[i]; + out_ptr++; + } + } + } + return (int)((intptr_t)out_ptr - (intptr_t)out_buffer); +} + +void dtls_reset_cookie_secret() { + tls_random(dtls_secret, sizeof(dtls_secret)); +} + +void tls_init() { + if (dependecies_loaded) + return; + DEBUG_PRINT("Initializing dependencies\n"); + dependecies_loaded = 1; +#ifdef LTM_DESC + ltc_mp = ltm_desc; +#else +#ifdef TFM_DESC + ltc_mp = tfm_desc; +#else +#ifdef GMP_DESC + ltc_mp = gmp_desc; +#endif +#endif +#endif + register_prng(&sprng_desc); + register_hash(&sha256_desc); + register_hash(&sha1_desc); + register_hash(&sha384_desc); + register_hash(&sha512_desc); + register_hash(&md5_desc); + register_cipher(&aes_desc); +#ifdef TLS_FORWARD_SECRECY + init_curves(); +#endif + dtls_reset_cookie_secret(); +} + +#ifdef TLS_FORWARD_SECRECY +int _private_tls_dh_shared_secret(DHKey *private_key, DHKey *public_key, unsigned char *out, unsigned long *outlen) { + void *tmp; + unsigned long x; + int err; + + if ((!private_key) || (!public_key) || (!out) || (!outlen)) + return TLS_GENERIC_ERROR; + + /* compute y^x mod p */ + if ((err = mp_init(&tmp)) != CRYPT_OK) + return err; + + if ((err = mp_exptmod(public_key->y, private_key->x, private_key->p, tmp)) != CRYPT_OK) { + mp_clear(tmp); + return err; + } + + x = (unsigned long)mp_unsigned_bin_size(tmp); + if (*outlen < x) { + err = CRYPT_BUFFER_OVERFLOW; + mp_clear(tmp); + return err; + } + + if ((err = mp_to_unsigned_bin(tmp, out)) != CRYPT_OK) { + mp_clear(tmp); + return err; + } + *outlen = x; + mp_clear(tmp); + return 0; +} + +unsigned char *_private_tls_decrypt_dhe(struct TLSContext *context, const unsigned char *buffer, unsigned int len, unsigned int *size, int clear_key) { + *size = 0; + if ((!len) || (!context) || (!context->dhe)) { + DEBUG_PRINT("No private DHE key set\n"); + return NULL; + } + + unsigned long out_size = len; + void *Yc = NULL; + + if (mp_init(&Yc)) { + DEBUG_PRINT("ERROR CREATING Yc\n"); + return NULL; + } + if (mp_read_unsigned_bin(Yc, (unsigned char *)buffer, len)) { + DEBUG_PRINT("ERROR LOADING DHE Yc\n"); + mp_clear(Yc); + return NULL; + } + + unsigned char *out = (unsigned char *)TLS_MALLOC(len); + DHKey client_key; + memset(&client_key, 0, sizeof(DHKey)); + + client_key.p = context->dhe->p; + client_key.g = context->dhe->g; + client_key.y = Yc; + int err = _private_tls_dh_shared_secret(context->dhe, &client_key, out, &out_size); + // don't delete p and g + client_key.p = NULL; + client_key.g = NULL; + _private_tls_dh_clear_key(&client_key); + // not needing the dhe key anymore + if (clear_key) + _private_tls_dhe_free(context); + if (err) { + DEBUG_PRINT("DHE DECRYPT ERROR %i\n", err); + TLS_FREE(out); + return NULL; + } + DEBUG_PRINT("OUT_SIZE: %lu\n", out_size); + DEBUG_DUMP_HEX_LABEL("DHE", out, out_size); + *size = (unsigned int)out_size; + return out; +} + +unsigned char *_private_tls_decrypt_ecc_dhe(struct TLSContext *context, const unsigned char *buffer, unsigned int len, unsigned int *size, int clear_key) { + *size = 0; + if ((!len) || (!context) || (!context->ecc_dhe)) { + DEBUG_PRINT("No private ECC DHE key set\n"); + return NULL; + } + + const struct ECCCurveParameters *curve; + if (context->curve) + curve = context->curve; + else + curve = default_curve; + + ltc_ecc_set_type *dp = (ltc_ecc_set_type *)&curve->dp; + + ecc_key client_key; + memset(&client_key, 0, sizeof(client_key)); + if (ecc_ansi_x963_import_ex(buffer, len, &client_key, dp)) { + DEBUG_PRINT("Error importing ECC DHE key\n"); + return NULL; + } + unsigned char *out = (unsigned char *)TLS_MALLOC(len); + unsigned long out_size = len; + + int err = ecc_shared_secret(context->ecc_dhe, &client_key, out, &out_size); + ecc_free(&client_key); + if (clear_key) + _private_tls_ecc_dhe_free(context); + if (err) { + DEBUG_PRINT("ECC DHE DECRYPT ERROR %i\n", err); + TLS_FREE(out); + return NULL; + } + DEBUG_PRINT("OUT_SIZE: %lu\n", out_size); + DEBUG_DUMP_HEX_LABEL("ECC DHE", out, out_size); + *size = (unsigned int)out_size; + return out; +} +#endif + +unsigned char *_private_tls_decrypt_rsa(struct TLSContext *context, const unsigned char *buffer, unsigned int len, unsigned int *size) { + *size = 0; + if ((!len) || (!context) || (!context->private_key) || (!context->private_key->der_bytes) || (!context->private_key->der_len)) { + DEBUG_PRINT("No private key set\n"); + return NULL; + } + tls_init(); + rsa_key key; + int err; + err = rsa_import(context->private_key->der_bytes, context->private_key->der_len, &key); + + if (err) { + DEBUG_PRINT("Error importing RSA key (code: %i)\n", err); + return NULL; + } + unsigned char *out = (unsigned char *)TLS_MALLOC(len); + unsigned long out_size = len; + int res = 0; + +#if (CRYPT >= 0x0118) + err = rsa_decrypt_key_ex(buffer, len, out, &out_size, NULL, 0, -1, -1, LTC_PKCS_1_V1_5, &res, &key); +#else + err = rsa_decrypt_key_ex(buffer, len, out, &out_size, NULL, 0, -1, LTC_PKCS_1_V1_5, &res, &key); +#endif + rsa_free(&key); + + if ((err) || (out_size != 48) || (ntohs(*(unsigned short *)out) != context->version)) { + // generate a random secret and continue (ROBOT fix) + // silently ignore and generate a random secret + out_size = 48; + tls_random(out, out_size); + *(unsigned short *)out = htons(context->version); + } + *size = (unsigned int)out_size; + return out; +} + +unsigned char *_private_tls_encrypt_rsa(struct TLSContext *context, const unsigned char *buffer, unsigned int len, unsigned int *size) { + *size = 0; + if ((!len) || (!context) || (!context->certificates) || (!context->certificates_count) || (!context->certificates[0]) || + (!context->certificates[0]->der_bytes) || (!context->certificates[0]->der_len)) { + DEBUG_PRINT("No certificate set\n"); + return NULL; + } + tls_init(); + rsa_key key; + int err; + err = rsa_import(context->certificates[0]->der_bytes, context->certificates[0]->der_len, &key); + + if (err) { + DEBUG_PRINT("Error importing RSA certificate (code: %i)\n", err); + return NULL; + } + unsigned long out_size = TLS_MAX_RSA_KEY; + unsigned char *out = (unsigned char *)TLS_MALLOC(out_size); + int hash_idx = find_hash("sha256"); + int prng_idx = find_prng("sprng"); +#if (CRYPT >= 0x0118) + err = rsa_encrypt_key_ex(buffer, len, out, &out_size, (unsigned char *)"Concept", 7, NULL, prng_idx, hash_idx, -1, LTC_PKCS_1_V1_5, &key); +#else + err = rsa_encrypt_key_ex(buffer, len, out, &out_size, (unsigned char *)"Concept", 7, NULL, prng_idx, hash_idx, LTC_PKCS_1_V1_5, &key); +#endif + rsa_free(&key); + if ((err) || (!out_size)) { + TLS_FREE(out); + return NULL; + } + *size = (unsigned int)out_size; + return out; +} + +#ifdef TLS_LEGACY_SUPPORT +int _private_rsa_verify_hash_md5sha1(const unsigned char *sig, unsigned long siglen, unsigned char *hash, unsigned long hashlen, int *stat, rsa_key *key) { + unsigned long modulus_bitlen, modulus_bytelen, x; + int err; + unsigned char *tmpbuf = NULL; + + if ((hash == NULL) || (sig == NULL) || (stat == NULL) || (key == NULL) || (!siglen) || (!hashlen)) + return TLS_GENERIC_ERROR; + + *stat = 0; + + modulus_bitlen = mp_count_bits((key->N)); + + modulus_bytelen = mp_unsigned_bin_size((key->N)); + if (modulus_bytelen != siglen) + return TLS_GENERIC_ERROR; + + tmpbuf = (unsigned char *)TLS_MALLOC(siglen); + if (!tmpbuf) + return TLS_GENERIC_ERROR; + + x = siglen; + if ((err = ltc_mp.rsa_me(sig, siglen, tmpbuf, &x, PK_PUBLIC, key)) != CRYPT_OK) { + TLS_FREE(tmpbuf); + return err; + } + + if (x != siglen) { + TLS_FREE(tmpbuf); + return CRYPT_INVALID_PACKET; + } + unsigned long out_len = siglen; + unsigned char *out = (unsigned char *)TLS_MALLOC(siglen); + if (!out) { + TLS_FREE(tmpbuf); + return TLS_GENERIC_ERROR; + } + + int decoded = 0; + err = pkcs_1_v1_5_decode(tmpbuf, x, LTC_PKCS_1_EMSA, modulus_bitlen, out, &out_len, &decoded); + if (decoded) { + if (out_len == hashlen) { + if (!memcmp(out, hash, hashlen)) + *stat = 1; + } + } + + TLS_FREE(tmpbuf); + TLS_FREE(out); + return err; +} +#endif + +int _private_tls_verify_rsa(struct TLSContext *context, unsigned int hash_type, const unsigned char *buffer, unsigned int len, const unsigned char *message, unsigned int message_len) { + tls_init(); + rsa_key key; + int err; + + if (context->is_server) { + if ((!len) || (!context->client_certificates) || (!context->client_certificates_count) || (!context->client_certificates[0]) || + (!context->client_certificates[0]->der_bytes) || (!context->client_certificates[0]->der_len)) { + DEBUG_PRINT("No client certificate set\n"); + return TLS_GENERIC_ERROR; + } + err = rsa_import(context->client_certificates[0]->der_bytes, context->client_certificates[0]->der_len, &key); + } else { + if ((!len) || (!context->certificates) || (!context->certificates_count) || (!context->certificates[0]) || + (!context->certificates[0]->der_bytes) || (!context->certificates[0]->der_len)) { + DEBUG_PRINT("No server certificate set\n"); + return TLS_GENERIC_ERROR; + } + err = rsa_import(context->certificates[0]->der_bytes, context->certificates[0]->der_len, &key); + } + if (err) { + DEBUG_PRINT("Error importing RSA certificate (code: %i)\n", err); + return TLS_GENERIC_ERROR; + } + int hash_idx = -1; + unsigned char hash[TLS_MAX_HASH_LEN]; + unsigned int hash_len = 0; + hash_state state; + switch (hash_type) { + case md5: + hash_idx = find_hash("md5"); + err = md5_init(&state); + TLS_ERROR(err, break); + err = md5_process(&state, message, message_len); + TLS_ERROR(err, break); + err = md5_done(&state, hash); + TLS_ERROR(err, break); + hash_len = 16; + break; + case sha1: + hash_idx = find_hash("sha1"); + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 20; + break; + case sha256: + hash_idx = find_hash("sha256"); + err = sha256_init(&state); + TLS_ERROR(err, break) + err = sha256_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha256_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 32; + break; + case sha384: + hash_idx = find_hash("sha384"); + err = sha384_init(&state); + TLS_ERROR(err, break) + err = sha384_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha384_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 48; + break; + case sha512: + hash_idx = find_hash("sha512"); + err = sha512_init(&state); + TLS_ERROR(err, break) + err = sha512_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha512_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 64; + break; +#ifdef TLS_LEGACY_SUPPORT + case _md5_sha1: + hash_idx = find_hash("md5"); + err = md5_init(&state); + TLS_ERROR(err, break) + err = md5_process(&state, message, message_len); + TLS_ERROR(err, break) + err = md5_done(&state, hash); + TLS_ERROR(err, break) + hash_idx = find_hash("sha1"); + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash + 16); + TLS_ERROR(err, break) + hash_len = 36; + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash + 16); + TLS_ERROR(err, break) + hash_len = 36; + break; +#endif + } + if ((hash_idx < 0) || (err)) { + DEBUG_PRINT("Unsupported hash type: %i\n", hash_type); + return TLS_GENERIC_ERROR; + } + int rsa_stat = 0; +#ifdef TLS_LEGACY_SUPPORT + if (hash_type == _md5_sha1) + err = _private_rsa_verify_hash_md5sha1(buffer, len, hash, hash_len, &rsa_stat, &key); + else +#endif +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + err = rsa_verify_hash_ex(buffer, len, hash, hash_len, LTC_PKCS_1_PSS, hash_idx, hash_len, &rsa_stat, &key); + else +#endif + err = rsa_verify_hash_ex(buffer, len, hash, hash_len, LTC_PKCS_1_V1_5, hash_idx, 0, &rsa_stat, &key); + rsa_free(&key); + if (err) + return 0; + return rsa_stat; +} + +#ifdef TLS_LEGACY_SUPPORT +int _private_rsa_sign_hash_md5sha1(const unsigned char *in, unsigned long inlen, unsigned char *out, unsigned long *outlen, rsa_key *key) { + unsigned long modulus_bitlen, modulus_bytelen, x; + int err; + + if ((in == NULL) || (out == NULL) || (outlen == NULL) || (key == NULL)) + return TLS_GENERIC_ERROR; + + modulus_bitlen = mp_count_bits((key->N)); + + modulus_bytelen = mp_unsigned_bin_size((key->N)); + if (modulus_bytelen > *outlen) { + *outlen = modulus_bytelen; + return CRYPT_BUFFER_OVERFLOW; + } + x = modulus_bytelen; + err = pkcs_1_v1_5_encode(in, inlen, LTC_PKCS_1_EMSA, modulus_bitlen, NULL, 0, out, &x); + if (err != CRYPT_OK) + return err; + + return ltc_mp.rsa_me(out, x, out, outlen, PK_PRIVATE, key); +} +#endif + +int _private_tls_sign_rsa(struct TLSContext *context, unsigned int hash_type, const unsigned char *message, unsigned int message_len, unsigned char *out, unsigned long *outlen) { + if ((!outlen) || (!context) || (!out) || (!outlen) || (!context->private_key) || (!context->private_key->der_bytes) || (!context->private_key->der_len)) { + DEBUG_PRINT("No private key set\n"); + return TLS_GENERIC_ERROR; + } + tls_init(); + rsa_key key; + int err; + err = rsa_import(context->private_key->der_bytes, context->private_key->der_len, &key); + + if (err) { + DEBUG_PRINT("Error importing RSA certificate (code: %i)\n", err); + return TLS_GENERIC_ERROR; + } + int hash_idx = -1; + unsigned char hash[TLS_MAX_HASH_LEN]; + unsigned int hash_len = 0; + hash_state state; + switch (hash_type) { + case md5: + hash_idx = find_hash("md5"); + err = md5_init(&state); + TLS_ERROR(err, break) + err = md5_process(&state, message, message_len); + TLS_ERROR(err, break) + err = md5_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 16; + break; + case sha1: + hash_idx = find_hash("sha1"); + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 20; + break; + case sha256: + hash_idx = find_hash("sha256"); + err = sha256_init(&state); + TLS_ERROR(err, break) + err = sha256_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha256_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 32; + break; + case sha384: + hash_idx = find_hash("sha384"); + err = sha384_init(&state); + TLS_ERROR(err, break) + err = sha384_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha384_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 48; + break; + case sha512: + hash_idx = find_hash("sha512"); + err = sha512_init(&state); + TLS_ERROR(err, break) + err = sha512_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha512_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 64; + break; + case _md5_sha1: + hash_idx = find_hash("md5"); + err = md5_init(&state); + TLS_ERROR(err, break) + err = md5_process(&state, message, message_len); + TLS_ERROR(err, break) + err = md5_done(&state, hash); + TLS_ERROR(err, break) + hash_idx = find_hash("sha1"); + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash + 16); + TLS_ERROR(err, break) + hash_len = 36; + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash + 16); + TLS_ERROR(err, break) + hash_len = 36; + break; + } +#ifdef TLS_LEGACY_SUPPORT + if (hash_type == _md5_sha1) { + if (err) { + DEBUG_PRINT("Unsupported hash type: %i\n", hash_type); + return TLS_GENERIC_ERROR; + } + err = _private_rsa_sign_hash_md5sha1(hash, hash_len, out, outlen, &key); + } else +#endif + { + if ((hash_idx < 0) || (err)) { + DEBUG_PRINT("Unsupported hash type: %i\n", hash_type); + return TLS_GENERIC_ERROR; + } +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + err = rsa_sign_hash_ex(hash, hash_len, out, outlen, LTC_PKCS_1_PSS, NULL, find_prng("sprng"), hash_idx, hash_type == sha256 ? 32 : 48, &key); + else +#endif + err = rsa_sign_hash_ex(hash, hash_len, out, outlen, LTC_PKCS_1_V1_5, NULL, find_prng("sprng"), hash_idx, 0, &key); + } + rsa_free(&key); + if (err) + return 0; + + return 1; +} + +#ifdef TLS_ECDSA_SUPPORTED + +#if CRYPT >= 0x0118 + +int _private_tls_is_point(ecc_key *key) { + void *prime, *a, *b, *t1, *t2; + int err; + + void *x = key->pubkey.x; + void *y = key->pubkey.y; + + prime = key->dp.prime; + b = key->dp.B; + a = key->dp.A; + + if ((err = mp_init_multi(&t1, &t2, NULL)) != CRYPT_OK) + return err; + + /* compute y^2 */ + if ((err = mp_sqr(y, t1)) != CRYPT_OK) + goto cleanup; + + /* compute x^3 */ + if ((err = mp_sqr(x, t2)) != CRYPT_OK) + goto cleanup; + if ((err = mp_mod(t2, prime, t2)) != CRYPT_OK) + goto cleanup; + if ((err = mp_mul(x, t2, t2)) != CRYPT_OK) + goto cleanup; + + /* compute y^2 - x^3 */ + if ((err = mp_sub(t1, t2, t1)) != CRYPT_OK) + goto cleanup; + + /* compute y^2 - x^3 - a*x */ + if ((err = mp_submod(prime, a, prime, t2)) != CRYPT_OK) + goto cleanup; + if ((err = mp_mulmod(t2, x, prime, t2)) != CRYPT_OK) + goto cleanup; + if ((err = mp_addmod(t1, t2, prime, t1)) != CRYPT_OK) + goto cleanup; + + /* adjust range (0, prime) */ + while (mp_cmp_d(t1, 0) == LTC_MP_LT) { + if ((err = mp_add(t1, prime, t1)) != CRYPT_OK) + goto cleanup; + } + while (mp_cmp(t1, prime) != LTC_MP_LT) { + if ((err = mp_sub(t1, prime, t1)) != CRYPT_OK) + goto cleanup; + } + + /* compare to b */ + if (mp_cmp(t1, b) != LTC_MP_EQ) { + err = CRYPT_INVALID_PACKET; + } else { + err = CRYPT_OK; + } + +cleanup: + mp_clear_multi(t1, t2, NULL); + return err; +} + +#else + +static int _private_tls_is_point(ecc_key *key) { + void *prime, *b, *t1, *t2; + int err; + + if ((err = mp_init_multi(&prime, &b, &t1, &t2, NULL)) != CRYPT_OK) { + return err; + } + + /* load prime and b */ + if ((err = mp_read_radix(prime, (const char *)TLS_TOMCRYPT_PRIVATE_DP(key)->prime, 16)) != CRYPT_OK) { + goto error; + } + if ((err = mp_read_radix(b, (const char *)TLS_TOMCRYPT_PRIVATE_DP(key)->B, 16)) != CRYPT_OK) { + goto error; + } + + /* compute y^2 */ + if ((err = mp_sqr(key->pubkey.y, t1)) != CRYPT_OK) { + goto error; + } + + /* compute x^3 */ + if ((err = mp_sqr(key->pubkey.x, t2)) != CRYPT_OK) { + goto error; + } + + if ((err = mp_mod(t2, prime, t2)) != CRYPT_OK) { + goto error; + } + + if ((err = mp_mul(key->pubkey.x, t2, t2)) != CRYPT_OK) { + goto error; + } + + /* compute y^2 - x^3 */ + if ((err = mp_sub(t1, t2, t1)) != CRYPT_OK) { + goto error; + } + + /* compute y^2 - x^3 + 3x */ + if ((err = mp_add(t1, key->pubkey.x, t1)) != CRYPT_OK) { + goto error; + } + if ((err = mp_add(t1, key->pubkey.x, t1)) != CRYPT_OK) { + goto error; + } + if ((err = mp_add(t1, key->pubkey.x, t1)) != CRYPT_OK) { + goto error; + } + if ((err = mp_mod(t1, prime, t1)) != CRYPT_OK) { + goto error; + } + while (mp_cmp_d(t1, 0) == LTC_MP_LT) { + if ((err = mp_add(t1, prime, t1)) != CRYPT_OK) { + goto error; + } + } + while (mp_cmp(t1, prime) != LTC_MP_LT) { + if ((err = mp_sub(t1, prime, t1)) != CRYPT_OK) { + goto error; + } + } + + /* compare to b */ + if (mp_cmp(t1, b) != LTC_MP_EQ) { + err = CRYPT_INVALID_PACKET; + } else { + err = CRYPT_OK; + } + +error: + mp_clear_multi(prime, b, t1, t2, NULL); + return err; +} + +#endif + +int _private_tls_ecc_import_key(const unsigned char *private_key, int private_len, const unsigned char *public_key, int public_len, ecc_key *key, const ltc_ecc_set_type *dp) { + int err; + + if ((!key) || (!ltc_mp.name)) + return CRYPT_MEM; + + key->type = PK_PRIVATE; + +#if CRYPT >= 0x0118 + if ((err = ecc_set_curve(dp, key)) != CRYPT_OK) + return err; +#else + if (mp_init_multi(&key->pubkey.x, &key->pubkey.y, &key->pubkey.z, &key->k, NULL) != CRYPT_OK) + return CRYPT_MEM; +#endif + + if ((public_len) && (!public_key[0])) { + public_key++; + public_len--; + } + if ((err = mp_read_unsigned_bin(key->pubkey.x, (unsigned char *)public_key + 1, (public_len - 1) >> 1)) != CRYPT_OK) { + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + + if ((err = mp_read_unsigned_bin(key->pubkey.y, (unsigned char *)public_key + 1 + ((public_len - 1) >> 1), (public_len - 1) >> 1)) != CRYPT_OK) { + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + + if ((err = mp_read_unsigned_bin(key->k, (unsigned char *)private_key, private_len)) != CRYPT_OK) { + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + +#if CRYPT < 0x0118 + TLS_TOMCRYPT_PRIVATE_SET_INDEX(key, -1); + TLS_TOMCRYPT_PRIVATE_DP(key) = dp; +#endif + + /* set z */ + if ((err = mp_set(key->pubkey.z, 1)) != CRYPT_OK) { + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + + /* is it a point on the curve? */ + if ((err = _private_tls_is_point(key)) != CRYPT_OK) { + DEBUG_PRINT("KEY IS NOT ON CURVE\n"); + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + + /* we're good */ + return CRYPT_OK; +} + +int _private_tls_sign_ecdsa(struct TLSContext *context, unsigned int hash_type, const unsigned char *message, unsigned int message_len, unsigned char *out, unsigned long *outlen) { + if ((!outlen) || (!context) || (!out) || (!outlen) || (!context->ec_private_key) || + (!context->ec_private_key->priv) || (!context->ec_private_key->priv_len) || (!context->ec_private_key->pk) || (!context->ec_private_key->pk_len)) { + DEBUG_PRINT("No private ECDSA key set\n"); + return TLS_GENERIC_ERROR; + } + + const struct ECCCurveParameters *curve = NULL; + + switch (context->ec_private_key->ec_algorithm) { + case 19: + curve = &secp192r1; + break; + case 20: + curve = &secp224k1; + break; + case 21: + curve = &secp224r1; + break; + case 22: + curve = &secp256k1; + break; + case 23: + curve = &secp256r1; + break; + case 24: + curve = &secp384r1; + break; + case 25: + curve = &secp521r1; + break; + default: + DEBUG_PRINT("UNSUPPORTED CURVE\n"); + } + + if (!curve) + return TLS_GENERIC_ERROR; + + tls_init(); + ecc_key key; + int err; + memset(&key, 0, sizeof(key)); + ltc_ecc_set_type *dp = (ltc_ecc_set_type *)&curve->dp; + + // broken ... fix this + err = _private_tls_ecc_import_key(context->ec_private_key->priv, context->ec_private_key->priv_len, context->ec_private_key->pk, context->ec_private_key->pk_len, &key, dp); + if (err) { + DEBUG_PRINT("Error importing ECC certificate (code: %i)\n", (int)err); + return TLS_GENERIC_ERROR; + } + unsigned char hash[TLS_MAX_HASH_LEN]; + unsigned int hash_len = 0; + hash_state state; + switch (hash_type) { + case md5: + err = md5_init(&state); + TLS_ERROR(err, break) + err = md5_process(&state, message, message_len); + TLS_ERROR(err, break) + err = md5_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 16; + break; + case sha1: + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 20; + break; + case sha256: + err = sha256_init(&state); + TLS_ERROR(err, break) + err = sha256_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha256_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 32; + break; + case sha384: + err = sha384_init(&state); + TLS_ERROR(err, break) + err = sha384_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha384_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 48; + break; + case sha512: + err = sha512_init(&state); + TLS_ERROR(err, break) + err = sha512_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha512_done(&state, hash); + TLS_ERROR(err, break) + hash_len = 64; + break; + case _md5_sha1: + err = md5_init(&state); + TLS_ERROR(err, break) + err = md5_process(&state, message, message_len); + TLS_ERROR(err, break) + err = md5_done(&state, hash); + TLS_ERROR(err, break) + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash + 16); + TLS_ERROR(err, break) + hash_len = 36; + err = sha1_init(&state); + TLS_ERROR(err, break) + err = sha1_process(&state, message, message_len); + TLS_ERROR(err, break) + err = sha1_done(&state, hash + 16); + TLS_ERROR(err, break) + hash_len = 36; + break; + } + + if (err) { + DEBUG_PRINT("Unsupported hash type: %i\n", hash_type); + return TLS_GENERIC_ERROR; + } + + // "Let z be the Ln leftmost bits of e, where Ln is the bit length of the group order n." + if (hash_len > (unsigned int)curve->size) + hash_len = (unsigned int)curve->size; + + err = ecc_sign_hash(hash, hash_len, out, outlen, NULL, find_prng("sprng"), &key); + DEBUG_DUMP_HEX_LABEL("ECC SIGNATURE", out, *outlen); + ecc_free(&key); + if (err) + return 0; + + return 1; +} + +#if defined(TLS_CLIENT_ECDSA) || defined(WITH_TLS_13) +int _private_tls_ecc_import_pk(const unsigned char *public_key, int public_len, ecc_key *key, const ltc_ecc_set_type *dp) { + int err; + + if ((!key) || (!ltc_mp.name)) + return CRYPT_MEM; + + key->type = PK_PUBLIC; + +#if CRYPT >= 0x0118 + if ((err = ecc_set_curve(dp, key)) != CRYPT_OK) + return err; +#else + if (mp_init_multi(&key->pubkey.x, &key->pubkey.y, &key->pubkey.z, &key->k, NULL) != CRYPT_OK) + return CRYPT_MEM; +#endif + + if ((public_len) && (!public_key[0])) { + public_key++; + public_len--; + } + if ((err = mp_read_unsigned_bin(key->pubkey.x, (unsigned char *)public_key + 1, (public_len - 1) >> 1)) != CRYPT_OK) { + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + + if ((err = mp_read_unsigned_bin(key->pubkey.y, (unsigned char *)public_key + 1 + ((public_len - 1) >> 1), (public_len - 1) >> 1)) != CRYPT_OK) { + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + +#if CRYPT < 0x0118 + TLS_TOMCRYPT_PRIVATE_SET_INDEX(key, -1); + TLS_TOMCRYPT_PRIVATE_DP(key) = dp; +#endif + + /* set z */ + if ((err = mp_set(key->pubkey.z, 1)) != CRYPT_OK) { + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + + /* is it a point on the curve? */ + if ((err = _private_tls_is_point(key)) != CRYPT_OK) { + DEBUG_PRINT("KEY IS NOT ON CURVE\n"); + mp_clear_multi(key->pubkey.x, key->pubkey.y, key->pubkey.z, key->k, NULL); + return err; + } + + /* we're good */ + return CRYPT_OK; +} + +int _private_tls_verify_ecdsa(struct TLSContext *context, unsigned int hash_type, const unsigned char *buffer, unsigned int len, const unsigned char *message, unsigned int message_len, const struct ECCCurveParameters *curve_hint) { + tls_init(); + ecc_key key; + int err; + + if (!curve_hint) + curve_hint = context->curve; + + if (context->is_server) { + if ((!len) || (!context->client_certificates) || (!context->client_certificates_count) || (!context->client_certificates[0]) || + (!context->client_certificates[0]->pk) || (!context->client_certificates[0]->pk_len) || (!curve_hint)) { + DEBUG_PRINT("No client certificate set\n"); + return TLS_GENERIC_ERROR; + } + err = _private_tls_ecc_import_pk(context->client_certificates[0]->pk, context->client_certificates[0]->pk_len, &key, (ltc_ecc_set_type *)&curve_hint->dp); + } else { + if ((!len) || (!context->certificates) || (!context->certificates_count) || (!context->certificates[0]) || + (!context->certificates[0]->pk) || (!context->certificates[0]->pk_len) || (!curve_hint)) { + DEBUG_PRINT("No server certificate set\n"); + return TLS_GENERIC_ERROR; + } + err = _private_tls_ecc_import_pk(context->certificates[0]->pk, context->certificates[0]->pk_len, &key, (ltc_ecc_set_type *)&curve_hint->dp); + } + if (err) { + DEBUG_PRINT("Error importing ECC certificate (code: %i)", err); + return TLS_GENERIC_ERROR; + } + int hash_idx = -1; + unsigned char hash[TLS_MAX_HASH_LEN]; + unsigned int hash_len = 0; + hash_state state; + switch (hash_type) { + case md5: + hash_idx = find_hash("md5"); + err = md5_init(&state); + if (!err) { + err = md5_process(&state, message, message_len); + if (!err) + err = md5_done(&state, hash); + } + hash_len = 16; + break; + case sha1: + hash_idx = find_hash("sha1"); + err = sha1_init(&state); + if (!err) { + err = sha1_process(&state, message, message_len); + if (!err) + err = sha1_done(&state, hash); + } + hash_len = 20; + break; + case sha256: + hash_idx = find_hash("sha256"); + err = sha256_init(&state); + if (!err) { + err = sha256_process(&state, message, message_len); + if (!err) + err = sha256_done(&state, hash); + } + hash_len = 32; + break; + case sha384: + hash_idx = find_hash("sha384"); + err = sha384_init(&state); + if (!err) { + err = sha384_process(&state, message, message_len); + if (!err) + err = sha384_done(&state, hash); + } + hash_len = 48; + break; + case sha512: + hash_idx = find_hash("sha512"); + err = sha512_init(&state); + if (!err) { + err = sha512_process(&state, message, message_len); + if (!err) + err = sha512_done(&state, hash); + } + hash_len = 64; + break; +#ifdef TLS_LEGACY_SUPPORT + case _md5_sha1: + hash_idx = find_hash("md5"); + err = md5_init(&state); + if (!err) { + err = md5_process(&state, message, message_len); + if (!err) + err = md5_done(&state, hash); + } + hash_idx = find_hash("sha1"); + err = sha1_init(&state); + if (!err) { + err = sha1_process(&state, message, message_len); + if (!err) + err = sha1_done(&state, hash + 16); + } + hash_len = 36; + err = sha1_init(&state); + if (!err) { + err = sha1_process(&state, message, message_len); + if (!err) + err = sha1_done(&state, hash + 16); + } + hash_len = 36; + break; +#endif + } + if ((hash_idx < 0) || (err)) { + DEBUG_PRINT("Unsupported hash type: %i\n", hash_type); + return TLS_GENERIC_ERROR; + } + int ecc_stat = 0; + err = ecc_verify_hash(buffer, len, hash, hash_len, &ecc_stat, &key); + ecc_free(&key); + if (err) + return 0; + return ecc_stat; +} +#endif + +#endif + +unsigned int _private_tls_random_int(int limit) { + unsigned int res = 0; + tls_random((unsigned char *)&res, sizeof(int)); + if (limit) + res %= limit; + return res; +} + +void _private_tls_sleep(unsigned int microseconds) { +#ifdef _WIN32 + Sleep(microseconds/1000); +#else + struct timespec ts; + + ts.tv_sec = (unsigned int) (microseconds / 1000000); + ts.tv_nsec = (unsigned int) (microseconds % 1000000) * 1000ul; + + nanosleep(&ts, NULL); +#endif +} + +void _private_random_sleep(struct TLSContext *context, int max_microseconds) { + if (context) + context->sleep_until = (unsigned int)time(NULL) + _private_tls_random_int(max_microseconds/1000000 * TLS_MAX_ERROR_IDLE_S); + else + _private_tls_sleep(_private_tls_random_int(max_microseconds)); +} + +void _private_tls_prf_helper(int hash_idx, unsigned long dlen, unsigned char *output, unsigned int outlen, const unsigned char *secret, const unsigned int secret_len, + const unsigned char *label, unsigned int label_len, unsigned char *seed, unsigned int seed_len, + unsigned char *seed_b, unsigned int seed_b_len) { + unsigned char digest_out0[TLS_MAX_HASH_LEN]; + unsigned char digest_out1[TLS_MAX_HASH_LEN]; + unsigned int i; + hmac_state hmac; + + hmac_init(&hmac, hash_idx, secret, secret_len); + hmac_process(&hmac, label, label_len); + + hmac_process(&hmac, seed, seed_len); + if ((seed_b) && (seed_b_len)) + hmac_process(&hmac, seed_b, seed_b_len); + hmac_done(&hmac, digest_out0, &dlen); + int idx = 0; + while (outlen) { + hmac_init(&hmac, hash_idx, secret, secret_len); + hmac_process(&hmac, digest_out0, dlen); + hmac_process(&hmac, label, label_len); + hmac_process(&hmac, seed, seed_len); + if ((seed_b) && (seed_b_len)) + hmac_process(&hmac, seed_b, seed_b_len); + hmac_done(&hmac, digest_out1, &dlen); + + unsigned int copylen = outlen; + if (copylen > dlen) + copylen = dlen; + + for (i = 0; i < copylen; i++) { + output[idx++] ^= digest_out1[i]; + outlen--; + } + + if (!outlen) + break; + + hmac_init(&hmac, hash_idx, secret, secret_len); + hmac_process(&hmac, digest_out0, dlen); + hmac_done(&hmac, digest_out0, &dlen); + } +} + +#ifdef WITH_TLS_13 +int _private_tls_hkdf_label(const char *label, unsigned char label_len, const unsigned char *data, unsigned char data_len, unsigned char *hkdflabel, unsigned short length, const char *prefix) { + *(unsigned short *)hkdflabel = htons(length); + int prefix_len; + if (prefix) { + prefix_len = (int)strlen(prefix); + memcpy(&hkdflabel[3], prefix, prefix_len); + } else { + memcpy(&hkdflabel[3], "tls13 ", 6); + prefix_len = 6; + } + hkdflabel[2] = (unsigned char)prefix_len + label_len; + memcpy(&hkdflabel[3 + prefix_len], label, label_len); + hkdflabel[3 + prefix_len + label_len] = (unsigned char)data_len; + if (data_len) + memcpy(&hkdflabel[4 + prefix_len + label_len], data, data_len); + return 4 + prefix_len + label_len + data_len; +} + +int _private_tls_hkdf_extract(unsigned int mac_length, unsigned char *output, unsigned int outlen, const unsigned char *salt, unsigned int salt_len, const unsigned char *ikm, unsigned char ikm_len) { + unsigned long dlen = outlen; + static unsigned char dummy_label[1] = { 0 }; + if ((!salt) || (salt_len == 0)) { + salt_len = 1; + salt = dummy_label; + } + int hash_idx; + if (mac_length == TLS_SHA384_MAC_SIZE) { + hash_idx = find_hash("sha384"); + dlen = mac_length; + } else + hash_idx = find_hash("sha256"); + + hmac_state hmac; + hmac_init(&hmac, hash_idx, salt, salt_len); + hmac_process(&hmac, ikm, ikm_len); + hmac_done(&hmac, output, &dlen); + DEBUG_DUMP_HEX_LABEL("EXTRACT", output, dlen); + return dlen; +} + +void _private_tls_hkdf_expand(unsigned int mac_length, unsigned char *output, unsigned int outlen, const unsigned char *secret, unsigned int secret_len, const unsigned char *info, unsigned char info_len) { + unsigned char digest_out[TLS_MAX_HASH_LEN]; + unsigned long dlen = 32; + int hash_idx; + if (mac_length == TLS_SHA384_MAC_SIZE) { + hash_idx = find_hash("sha384"); + dlen = mac_length; + } else + hash_idx = find_hash("sha256"); + unsigned int i; + unsigned int idx = 0; + hmac_state hmac; + unsigned char i2 = 0; + while (outlen) { + hmac_init(&hmac, hash_idx, secret, secret_len); + if (i2) + hmac_process(&hmac, digest_out, dlen); + if ((info) && (info_len)) + hmac_process(&hmac, info, info_len); + i2++; + hmac_process(&hmac, &i2, 1); + hmac_done(&hmac, digest_out, &dlen); + + unsigned int copylen = outlen; + if (copylen > dlen) + copylen = (unsigned int)dlen; + + for (i = 0; i < copylen; i++) { + output[idx++] = digest_out[i]; + outlen--; + } + + if (!outlen) + break; + } +} + +void _private_tls_hkdf_expand_label(unsigned int mac_length, unsigned char *output, unsigned int outlen, const unsigned char *secret, unsigned int secret_len, const char *label, unsigned char label_len, const unsigned char *data, unsigned char data_len) { + unsigned char hkdf_label[512]; + int len = _private_tls_hkdf_label(label, label_len, data, data_len, hkdf_label, outlen, NULL); + DEBUG_DUMP_HEX_LABEL("INFO", hkdf_label, len); + _private_tls_hkdf_expand(mac_length, output, outlen, secret, secret_len, hkdf_label, len); +} +#endif + +void _private_tls_prf(struct TLSContext *context, + unsigned char *output, unsigned int outlen, const unsigned char *secret, const unsigned int secret_len, + const unsigned char *label, unsigned int label_len, unsigned char *seed, unsigned int seed_len, + unsigned char *seed_b, unsigned int seed_b_len) { + if ((!secret) || (!secret_len)) { + DEBUG_PRINT("NULL SECRET\n"); + return; + } + if ((context->version != TLS_V12) && (context->version != DTLS_V12)) { + int md5_hash_idx = find_hash("md5"); + int sha1_hash_idx = find_hash("sha1"); + int half_secret = (secret_len + 1) / 2; + + memset(output, 0, outlen); + _private_tls_prf_helper(md5_hash_idx, 16, output, outlen, secret, half_secret, label, label_len, seed, seed_len, seed_b, seed_b_len); + _private_tls_prf_helper(sha1_hash_idx, 20, output, outlen, secret + (secret_len - half_secret), secret_len - half_secret, label, label_len, seed, seed_len, seed_b, seed_b_len); + } else { + // sha256_hmac + unsigned char digest_out0[TLS_MAX_HASH_LEN]; + unsigned char digest_out1[TLS_MAX_HASH_LEN]; + unsigned long dlen = 32; + int hash_idx; + unsigned int mac_length = _private_tls_mac_length(context); + if (mac_length == TLS_SHA384_MAC_SIZE) { + hash_idx = find_hash("sha384"); + dlen = mac_length; + } else + hash_idx = find_hash("sha256"); + unsigned int i; + hmac_state hmac; + + hmac_init(&hmac, hash_idx, secret, secret_len); + hmac_process(&hmac, label, label_len); + + hmac_process(&hmac, seed, seed_len); + if ((seed_b) && (seed_b_len)) + hmac_process(&hmac, seed_b, seed_b_len); + hmac_done(&hmac, digest_out0, &dlen); + int idx = 0; + while (outlen) { + hmac_init(&hmac, hash_idx, secret, secret_len); + hmac_process(&hmac, digest_out0, dlen); + hmac_process(&hmac, label, label_len); + hmac_process(&hmac, seed, seed_len); + if ((seed_b) && (seed_b_len)) + hmac_process(&hmac, seed_b, seed_b_len); + hmac_done(&hmac, digest_out1, &dlen); + + unsigned int copylen = outlen; + if (copylen > dlen) + copylen = (unsigned int)dlen; + + for (i = 0; i < copylen; i++) { + output[idx++] = digest_out1[i]; + outlen--; + } + + if (!outlen) + break; + + hmac_init(&hmac, hash_idx, secret, secret_len); + hmac_process(&hmac, digest_out0, dlen); + hmac_done(&hmac, digest_out0, &dlen); + } + } +} + +int _private_tls_key_length(struct TLSContext *context) { + switch (context->cipher) { + case TLS_RSA_WITH_AES_128_CBC_SHA: + case TLS_RSA_WITH_AES_128_CBC_SHA256: + case TLS_RSA_WITH_AES_128_GCM_SHA256: + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_AES_128_GCM_SHA256: + return 16; + case TLS_RSA_WITH_AES_256_CBC_SHA: + case TLS_RSA_WITH_AES_256_CBC_SHA256: + case TLS_RSA_WITH_AES_256_GCM_SHA384: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_AES_256_GCM_SHA384: + case TLS_CHACHA20_POLY1305_SHA256: + return 32; + } + return 0; +} + +int _private_tls_is_aead(struct TLSContext *context) { + switch (context->cipher) { + case TLS_RSA_WITH_AES_128_GCM_SHA256: + case TLS_RSA_WITH_AES_256_GCM_SHA384: + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_DHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: + case TLS_AES_128_GCM_SHA256: + case TLS_AES_256_GCM_SHA384: + return 1; + case TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_CHACHA20_POLY1305_SHA256: + return 2; + } + return 0; +} + + + +unsigned int _private_tls_mac_length(struct TLSContext *context) { + switch (context->cipher) { + case TLS_RSA_WITH_AES_128_CBC_SHA: + case TLS_RSA_WITH_AES_256_CBC_SHA: + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA: + return TLS_SHA1_MAC_SIZE; + case TLS_RSA_WITH_AES_128_CBC_SHA256: + case TLS_RSA_WITH_AES_256_CBC_SHA256: + case TLS_RSA_WITH_AES_128_GCM_SHA256: + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256: +#ifdef WITH_TLS_13 + case TLS_AES_128_GCM_SHA256: + case TLS_CHACHA20_POLY1305_SHA256: + case TLS_AES_128_CCM_SHA256: + case TLS_AES_128_CCM_8_SHA256: +#endif + return TLS_SHA256_MAC_SIZE; + case TLS_RSA_WITH_AES_256_GCM_SHA384: + case TLS_DHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: +#ifdef WITH_TLS_13 + case TLS_AES_256_GCM_SHA384: +#endif + return TLS_SHA384_MAC_SIZE; + } + return 0; +} + +#ifdef WITH_TLS_13 +int _private_tls13_key(struct TLSContext *context, int handshake) { + tls_init(); + + int key_length = _private_tls_key_length(context); + unsigned int mac_length = _private_tls_mac_length(context); + + if ((!context->premaster_key) || (!context->premaster_key_len)) + return 0; + + if ((!key_length) || (!mac_length)) { + DEBUG_PRINT("KEY EXPANSION FAILED, KEY LENGTH: %i, MAC LENGTH: %i\n", key_length, mac_length); + return 0; + } + + unsigned char *clientkey = NULL; + unsigned char *serverkey = NULL; + unsigned char *clientiv = NULL; + unsigned char *serveriv = NULL; + int is_aead = _private_tls_is_aead(context); + + unsigned char local_keybuffer[TLS_V13_MAX_KEY_SIZE]; + unsigned char local_ivbuffer[TLS_V13_MAX_IV_SIZE]; + unsigned char remote_keybuffer[TLS_V13_MAX_KEY_SIZE]; + unsigned char remote_ivbuffer[TLS_V13_MAX_IV_SIZE]; + + unsigned char prk[TLS_MAX_HASH_SIZE]; + unsigned char hash[TLS_MAX_HASH_SIZE]; + static unsigned char earlysecret[TLS_MAX_HASH_SIZE]; + + const char *server_key = "s ap traffic"; + const char *client_key = "c ap traffic"; + if (handshake) { + server_key = "s hs traffic"; + client_key = "c hs traffic"; + } + + unsigned char salt[TLS_MAX_HASH_SIZE]; + + hash_state md; + if (mac_length == TLS_SHA384_MAC_SIZE) { + sha384_init(&md); + sha384_done(&md, hash); + } else { + sha256_init(&md); + sha256_done(&md, hash); + } + // extract secret "early" + if ((context->master_key) && (context->master_key_len) && (!handshake)) { + DEBUG_DUMP_HEX_LABEL("USING PREVIOUS SECRET", context->master_key, context->master_key_len); + _private_tls_hkdf_expand_label(mac_length, salt, mac_length, context->master_key, context->master_key_len, "derived", 7, hash, mac_length); + DEBUG_DUMP_HEX_LABEL("salt", salt, mac_length); + _private_tls_hkdf_extract(mac_length, prk, mac_length, salt, mac_length, earlysecret, mac_length); + } else { + _private_tls_hkdf_extract(mac_length, prk, mac_length, NULL, 0, earlysecret, mac_length); + // derive secret for handshake "tls13 derived": + DEBUG_DUMP_HEX_LABEL("null hash", hash, mac_length); + _private_tls_hkdf_expand_label(mac_length, salt, mac_length, prk, mac_length, "derived", 7, hash, mac_length); + // extract secret "handshake": + DEBUG_DUMP_HEX_LABEL("salt", salt, mac_length); + _private_tls_hkdf_extract(mac_length, prk, mac_length, salt, mac_length, context->premaster_key, context->premaster_key_len); + } + + if (!is_aead) { + DEBUG_PRINT("KEY EXPANSION FAILED, NON AEAD CIPHER\n"); + return 0; + } + + unsigned char secret[TLS_MAX_MAC_SIZE]; + unsigned char hs_secret[TLS_MAX_HASH_SIZE]; + + int hash_size; + if (handshake) + hash_size = _private_tls_get_hash(context, hash); + else + hash_size = _private_tls_done_hash(context, hash); + DEBUG_DUMP_HEX_LABEL("messages hash", hash, hash_size); + + if (context->is_server) { + _private_tls_hkdf_expand_label(mac_length, hs_secret, mac_length, prk, mac_length, server_key, 12, context->server_finished_hash ? context->server_finished_hash : hash, hash_size); + DEBUG_DUMP_HEX_LABEL(server_key, hs_secret, mac_length); + serverkey = local_keybuffer; + serveriv = local_ivbuffer; + clientkey = remote_keybuffer; + clientiv = remote_ivbuffer; + } else { + _private_tls_hkdf_expand_label(mac_length, hs_secret, mac_length, prk, mac_length, client_key, 12, context->server_finished_hash ? context->server_finished_hash : hash, hash_size); + DEBUG_DUMP_HEX_LABEL(client_key, hs_secret, mac_length); + serverkey = remote_keybuffer; + serveriv = remote_ivbuffer; + clientkey = local_keybuffer; + clientiv = local_ivbuffer; + } + + int iv_length = TLS_13_AES_GCM_IV_LENGTH; +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) + iv_length = TLS_CHACHA20_IV_LENGTH; +#endif + + _private_tls_hkdf_expand_label(mac_length, local_keybuffer, key_length, hs_secret, mac_length, "key", 3, NULL, 0); + _private_tls_hkdf_expand_label(mac_length, local_ivbuffer, iv_length, hs_secret, mac_length, "iv", 2, NULL, 0); + if (context->is_server) + _private_tls_hkdf_expand_label(mac_length, secret, mac_length, prk, mac_length, client_key, 12, context->server_finished_hash ? context->server_finished_hash : hash, hash_size); + else + _private_tls_hkdf_expand_label(mac_length, secret, mac_length, prk, mac_length, server_key, 12, context->server_finished_hash ? context->server_finished_hash : hash, hash_size); + + _private_tls_hkdf_expand_label(mac_length, remote_keybuffer, key_length, secret, mac_length, "key", 3, NULL, 0); + _private_tls_hkdf_expand_label(mac_length, remote_ivbuffer, iv_length, secret, mac_length, "iv", 2, NULL, 0); + + DEBUG_DUMP_HEX_LABEL("CLIENT KEY", clientkey, key_length) + DEBUG_DUMP_HEX_LABEL("CLIENT IV", clientiv, iv_length) + DEBUG_DUMP_HEX_LABEL("SERVER KEY", serverkey, key_length) + DEBUG_DUMP_HEX_LABEL("SERVER IV", serveriv, iv_length) + + TLS_FREE(context->finished_key); + TLS_FREE(context->remote_finished_key); + if (handshake) { + context->finished_key = (unsigned char *)TLS_MALLOC(mac_length); + context->remote_finished_key = (unsigned char *)TLS_MALLOC(mac_length); + + if (context->finished_key) { + _private_tls_hkdf_expand_label(mac_length, context->finished_key, mac_length, hs_secret, mac_length, "finished", 8, NULL, 0); + DEBUG_DUMP_HEX_LABEL("FINISHED", context->finished_key, mac_length) + } + + if (context->remote_finished_key) { + _private_tls_hkdf_expand_label(mac_length, context->remote_finished_key, mac_length, secret, mac_length, "finished", 8, NULL, 0); + DEBUG_DUMP_HEX_LABEL("REMOTE FINISHED", context->remote_finished_key, mac_length) + } + } else { + context->finished_key = NULL; + context->remote_finished_key = NULL; + TLS_FREE(context->server_finished_hash); + context->server_finished_hash = NULL; + } + + if (context->is_server) { +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) { + memcpy(context->crypto.ctx_remote_mac.remote_nonce, clientiv, iv_length); + memcpy(context->crypto.ctx_local_mac.local_nonce, serveriv, iv_length); + } else +#endif + if (is_aead) { + memcpy(context->crypto.ctx_remote_mac.remote_iv, clientiv, iv_length); + memcpy(context->crypto.ctx_local_mac.local_iv, serveriv, iv_length); + } + if (_private_tls_crypto_create(context, key_length, serverkey, serveriv, clientkey, clientiv)) + return 0; + } else { +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) { + memcpy(context->crypto.ctx_local_mac.local_nonce, clientiv, iv_length); + memcpy(context->crypto.ctx_remote_mac.remote_nonce, serveriv, iv_length); + } else +#endif + if (is_aead) { + memcpy(context->crypto.ctx_local_mac.local_iv, clientiv, iv_length); + memcpy(context->crypto.ctx_remote_mac.remote_iv, serveriv, iv_length); + } + if (_private_tls_crypto_create(context, key_length, clientkey, clientiv, serverkey, serveriv)) + return 0; + } + context->crypto.created = 1 + is_aead; + if (context->exportable) { + TLS_FREE(context->exportable_keys); + context->exportable_keys = (unsigned char *)TLS_MALLOC(key_length * 2); + if (context->exportable_keys) { + if (context->is_server) { + memcpy(context->exportable_keys, serverkey, key_length); + memcpy(context->exportable_keys + key_length, clientkey, key_length); + } else { + memcpy(context->exportable_keys, clientkey, key_length); + memcpy(context->exportable_keys + key_length, serverkey, key_length); + } + context->exportable_size = key_length * 2; + } + } + TLS_FREE(context->master_key); + context->master_key = (unsigned char *)TLS_MALLOC(mac_length); + if (context->master_key) { + memcpy(context->master_key, prk, mac_length); + context->master_key_len = mac_length; + } + context->local_sequence_number = 0; + context->remote_sequence_number = 0; + + // extract client_mac_key(mac_key_length) + // extract server_mac_key(mac_key_length) + // extract client_key(enc_key_length) + // extract server_key(enc_key_length) + // extract client_iv(fixed_iv_lengh) + // extract server_iv(fixed_iv_length) + return 1; +} +#endif + +int _private_tls_expand_key(struct TLSContext *context) { + unsigned char key[TLS_MAX_KEY_EXPANSION_SIZE]; +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + return 0; +#endif + + if ((!context->master_key) || (!context->master_key_len)) + return 0; + + int key_length = _private_tls_key_length(context); + int mac_length = _private_tls_mac_length(context); + + if ((!key_length) || (!mac_length)) { + DEBUG_PRINT("KEY EXPANSION FAILED, KEY LENGTH: %i, MAC LENGTH: %i\n", key_length, mac_length); + return 0; + } + unsigned char *clientkey = NULL; + unsigned char *serverkey = NULL; + unsigned char *clientiv = NULL; + unsigned char *serveriv = NULL; + int iv_length = TLS_AES_IV_LENGTH; + int is_aead = _private_tls_is_aead(context); + if (context->is_server) + _private_tls_prf(context, key, sizeof(key), context->master_key, context->master_key_len, (unsigned char *)"key expansion", 13, context->local_random, TLS_SERVER_RANDOM_SIZE, context->remote_random, TLS_CLIENT_RANDOM_SIZE); + else + _private_tls_prf(context, key, sizeof(key), context->master_key, context->master_key_len, (unsigned char *)"key expansion", 13, context->remote_random, TLS_SERVER_RANDOM_SIZE, context->local_random, TLS_CLIENT_RANDOM_SIZE); + + DEBUG_DUMP_HEX_LABEL("LOCAL RANDOM ", context->local_random, TLS_SERVER_RANDOM_SIZE); + DEBUG_DUMP_HEX_LABEL("REMOTE RANDOM", context->remote_random, TLS_CLIENT_RANDOM_SIZE); + DEBUG_PRINT("\n=========== EXPANSION ===========\n"); + DEBUG_DUMP_HEX(key, TLS_MAX_KEY_EXPANSION_SIZE); + DEBUG_PRINT("\n"); + + int pos = 0; +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) { + iv_length = TLS_CHACHA20_IV_LENGTH; + } else +#endif + if (is_aead) { + iv_length = TLS_AES_GCM_IV_LENGTH; + } else { + if (context->is_server) { + memcpy(context->crypto.ctx_remote_mac.remote_mac, &key[pos], mac_length); + pos += mac_length; + memcpy(context->crypto.ctx_local_mac.local_mac, &key[pos], mac_length); + pos += mac_length; + } else { + memcpy(context->crypto.ctx_local_mac.local_mac, &key[pos], mac_length); + pos += mac_length; + memcpy(context->crypto.ctx_remote_mac.remote_mac, &key[pos], mac_length); + pos += mac_length; + } + } + + clientkey = &key[pos]; + pos += key_length; + serverkey = &key[pos]; + pos += key_length; + clientiv = &key[pos]; + pos += iv_length; + serveriv = &key[pos]; + pos += iv_length; + DEBUG_PRINT("EXPANSION %i/%i\n", (int)pos, (int)TLS_MAX_KEY_EXPANSION_SIZE); + DEBUG_DUMP_HEX_LABEL("CLIENT KEY", clientkey, key_length) + DEBUG_DUMP_HEX_LABEL("CLIENT IV", clientiv, iv_length) + DEBUG_DUMP_HEX_LABEL("CLIENT MAC KEY", context->is_server ? context->crypto.ctx_remote_mac.remote_mac : context->crypto.ctx_local_mac.local_mac, mac_length) + DEBUG_DUMP_HEX_LABEL("SERVER KEY", serverkey, key_length) + DEBUG_DUMP_HEX_LABEL("SERVER IV", serveriv, iv_length) + DEBUG_DUMP_HEX_LABEL("SERVER MAC KEY", context->is_server ? context->crypto.ctx_local_mac.local_mac : context->crypto.ctx_remote_mac.remote_mac, mac_length) + + if (context->is_server) { +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) { + memcpy(context->crypto.ctx_remote_mac.remote_nonce, clientiv, iv_length); + memcpy(context->crypto.ctx_local_mac.local_nonce, serveriv, iv_length); + } else +#endif + if (is_aead) { + memcpy(context->crypto.ctx_remote_mac.remote_aead_iv, clientiv, iv_length); + memcpy(context->crypto.ctx_local_mac.local_aead_iv, serveriv, iv_length); + } + if (_private_tls_crypto_create(context, key_length, serverkey, serveriv, clientkey, clientiv)) + return 0; + } else { +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) { + memcpy(context->crypto.ctx_local_mac.local_nonce, clientiv, iv_length); + memcpy(context->crypto.ctx_remote_mac.remote_nonce, serveriv, iv_length); + } else +#endif + if (is_aead) { + memcpy(context->crypto.ctx_local_mac.local_aead_iv, clientiv, iv_length); + memcpy(context->crypto.ctx_remote_mac.remote_aead_iv, serveriv, iv_length); + } + if (_private_tls_crypto_create(context, key_length, clientkey, clientiv, serverkey, serveriv)) + return 0; + } + + if (context->exportable) { + TLS_FREE(context->exportable_keys); + context->exportable_keys = (unsigned char *)TLS_MALLOC(key_length * 2); + if (context->exportable_keys) { + if (context->is_server) { + memcpy(context->exportable_keys, serverkey, key_length); + memcpy(context->exportable_keys + key_length, clientkey, key_length); + } else { + memcpy(context->exportable_keys, clientkey, key_length); + memcpy(context->exportable_keys + key_length, serverkey, key_length); + } + context->exportable_size = key_length * 2; + } + } + + // extract client_mac_key(mac_key_length) + // extract server_mac_key(mac_key_length) + // extract client_key(enc_key_length) + // extract server_key(enc_key_length) + // extract client_iv(fixed_iv_lengh) + // extract server_iv(fixed_iv_length) + return 1; +} + +int _private_tls_compute_key(struct TLSContext *context, unsigned int key_len) { +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + return 0; +#endif + if ((!context->premaster_key) || (!context->premaster_key_len) || (key_len < 48)) { + DEBUG_PRINT("CANNOT COMPUTE MASTER SECRET\n"); + return 0; + } + unsigned char master_secret_label[] = "master secret"; +#ifdef TLS_CHECK_PREMASTER_KEY + if (!tls_cipher_is_ephemeral(context)) { + unsigned short version = ntohs(*(unsigned short *)context->premaster_key); + // this check is not true for DHE/ECDHE ciphers + if (context->version > version) { + DEBUG_PRINT("Mismatch protocol version 0x(%x)\n", version); + return 0; + } + } +#endif + TLS_FREE(context->master_key); + context->master_key_len = 0; + context->master_key = NULL; + if ((context->version == TLS_V13) || (context->version == TLS_V12) || (context->version == TLS_V11) || (context->version == TLS_V10) || (context->version == DTLS_V13) || (context->version == DTLS_V12) || (context->version == DTLS_V10)) { + context->master_key = (unsigned char *)TLS_MALLOC(key_len); + if (!context->master_key) + return 0; + context->master_key_len = key_len; + if (context->is_server) { +#ifdef TLS_DTLS_EXTENDED_MASTER_SECRET + if ((context->dtls) && (context->dtls_data) && (context->dtls_data->extended_master_secret)) { + // master_secret = PRF(pre_master_secret, "extended master secret", session_hash); + unsigned char handshake_hash[TLS_MAX_SHA_SIZE]; + int hash_size = _private_tls_get_hash(context, handshake_hash); + DEBUG_DUMP_HEX_LABEL(">> HANDSHAKE HASH", handshake_hash, hash_size); + _private_tls_prf(context, + context->master_key, context->master_key_len, + context->premaster_key, context->premaster_key_len, + "extended master secret", 22, + handshake_hash, hash_size, + NULL, 0 + ); + } else +#endif + _private_tls_prf(context, + context->master_key, context->master_key_len, + context->premaster_key, context->premaster_key_len, + master_secret_label, 13, + context->remote_random, TLS_CLIENT_RANDOM_SIZE, + context->local_random, TLS_SERVER_RANDOM_SIZE + ); + } else { + _private_tls_prf(context, + context->master_key, context->master_key_len, + context->premaster_key, context->premaster_key_len, + master_secret_label, 13, + context->local_random, TLS_CLIENT_RANDOM_SIZE, + context->remote_random, TLS_SERVER_RANDOM_SIZE + ); + } + TLS_FREE(context->premaster_key); + context->premaster_key = NULL; + context->premaster_key_len = 0; + DEBUG_PRINT("\n=========== Master key ===========\n"); + DEBUG_DUMP_HEX(context->master_key, context->master_key_len); + DEBUG_PRINT("\n"); + _private_tls_expand_key(context); + return 1; + } + return 0; +} + +unsigned char *tls_pem_decode(const unsigned char *data_in, unsigned int input_length, int cert_index, unsigned int *output_len) { + unsigned int i; + *output_len = 0; + int alloc_len = input_length / 4 * 3; + unsigned char *output = (unsigned char *)TLS_MALLOC(alloc_len); + if (!output) + return NULL; + unsigned int start_at = 0; + unsigned int idx = 0; + for (i = 0; i < input_length; i++) { + if ((data_in[i] == '\n') || (data_in[i] == '\r')) + continue; + + if (data_in[i] != '-') { + // read entire line + while ((i < input_length) && (data_in[i] != '\n')) + i++; + continue; + } + + if (data_in[i] == '-') { + unsigned int end_idx = i; + //read until end of line + while ((i < input_length) && (data_in[i] != '\n')) + i++; + if (start_at) { + if (cert_index > 0) { + cert_index--; + start_at = 0; + } else { + idx = _private_b64_decode((const char *)&data_in[start_at], end_idx - start_at, output); + break; + } + } else + start_at = i + 1; + } + } + *output_len = idx; + if (!idx) { + TLS_FREE(output); + return NULL; + } + return output; +} + +int _is_oid(const unsigned char *oid, const unsigned char *compare_to, int compare_to_len) { + int i = 0; + while ((oid[i]) && (i < compare_to_len)) { + if (oid[i] != compare_to[i]) + return 0; + + i++; + } + return 1; +} + +int _is_oid2(const unsigned char *oid, const unsigned char *compare_to, int compare_to_len, int oid_len) { + int i = 0; + if (oid_len < compare_to_len) + compare_to_len = oid_len; + while (i < compare_to_len) { + if (oid[i] != compare_to[i]) + return 0; + + i++; + } + return 1; +} + +struct TLSCertificate *tls_create_certificate() { + struct TLSCertificate *cert = (struct TLSCertificate *)TLS_MALLOC(sizeof(struct TLSCertificate)); + if (cert) + memset(cert, 0, sizeof(struct TLSCertificate)); + return cert; +} + +int tls_certificate_valid_subject_name(const unsigned char *cert_subject, const char *subject) { + // no subjects ... + if (((!cert_subject) || (!cert_subject[0])) && ((!subject) || (!subject[0]))) + return 0; + + if ((!subject) || (!subject[0])) + return bad_certificate; + + if ((!cert_subject) || (!cert_subject[0])) + return bad_certificate; + + // exact match + if (!strcmp((const char *)cert_subject, subject)) + return 0; + + const char *wildcard = strchr((const char *)cert_subject, '*'); + if (wildcard) { + // 6.4.3 (1) The client SHOULD NOT attempt to match a presented identifier in + // which the wildcard character comprises a label other than the left-most label + if (!wildcard[1]) { + // subject is [*] + // or + // subject is [something*] .. invalid + return bad_certificate; + } + wildcard++; + const char *match = strstr(subject, wildcard); + if ((!match) && (wildcard[0] == '.')) { + // check *.domain.com agains domain.com + wildcard++; + if (!strcasecmp(subject, wildcard)) + return 0; + } + if (match) { + uintptr_t offset = (uintptr_t)match - (uintptr_t)subject; + if (offset) { + // check for foo.*.domain.com against *.domain.com (invalid) + if (memchr(subject, '.', offset)) + return bad_certificate; + } + // check if exact match + if (!strcasecmp(match, wildcard)) + return 0; + } + } + + return bad_certificate; +} + +int tls_certificate_valid_subject(struct TLSCertificate *cert, const char *subject) { + int i; + if (!cert) + return certificate_unknown; + int err = tls_certificate_valid_subject_name(cert->subject, subject); + if ((err) && (cert->san)) { + for (i = 0; i < cert->san_length; i++) { + err = tls_certificate_valid_subject_name(cert->san[i], subject); + if (!err) + return err; + } + } + return err; +} + +int tls_certificate_is_valid(struct TLSCertificate *cert) { + if (!cert) + return certificate_unknown; + if (!cert->not_before) + return certificate_unknown; + if (!cert->not_after) + return certificate_unknown; + //20160224182300Z// + char current_time[16]; + time_t t = time(NULL); + struct tm *utct = gmtime(&t); + if (utct) { + current_time[0] = 0; + snprintf(current_time, sizeof(current_time), "%04hu%02hhu%02hhu%02hhu%02hhu%02hhuZ", 1900 + utct->tm_year, utct->tm_mon + 1, utct->tm_mday, utct->tm_hour, utct->tm_min, utct->tm_sec); + if (strcasecmp((char *)cert->not_before, current_time) > 0) { + DEBUG_PRINT("Certificate is not yer valid, now: %s (validity: %s - %s)\n", current_time, cert->not_before, cert->not_after); + return certificate_expired; + } + if (strcasecmp((char *)cert->not_after, current_time) < 0) { + DEBUG_PRINT("Expired certificate, now: %s (validity: %s - %s)\n", current_time, cert->not_before, cert->not_after); + return certificate_expired; + } + DEBUG_PRINT("Valid certificate, now: %s (validity: %s - %s)\n", current_time, cert->not_before, cert->not_after); + } + return 0; +} + +void tls_certificate_set_copy(unsigned char **member, const unsigned char *val, int len) { + if (!member) + return; + TLS_FREE(*member); + if (len) { + *member = (unsigned char *)TLS_MALLOC(len + 1); + if (*member) { + memcpy(*member, val, len); + (*member)[len] = 0; + } + } else + *member = NULL; +} + +void tls_certificate_set_copy_date(unsigned char **member, const unsigned char *val, int len) { + if (!member) + return; + TLS_FREE(*member); + if (len > 4) { + *member = (unsigned char *)TLS_MALLOC(len + 3); + if (*member) { + if (val[0] == '9') { + (*member)[0]='1'; + (*member)[1]='9'; + } else { + (*member)[0]='2'; + (*member)[1]='0'; + } + memcpy(*member + 2, val, len); + (*member)[len] = 0; + } + } else + *member = NULL; +} + +void tls_certificate_set_key(struct TLSCertificate *cert, const unsigned char *val, int len) { + if ((!val[0]) && (len % 2)) { + val++; + len--; + } + tls_certificate_set_copy(&cert->pk, val, len); + if (cert->pk) + cert->pk_len = len; +} + +void tls_certificate_set_priv(struct TLSCertificate *cert, const unsigned char *val, int len) { + tls_certificate_set_copy(&cert->priv, val, len); + if (cert->priv) + cert->priv_len = len; +} + +void tls_certificate_set_sign_key(struct TLSCertificate *cert, const unsigned char *val, int len) { + if ((!val[0]) && (len % 2)) { + val++; + len--; + } + tls_certificate_set_copy(&cert->sign_key, val, len); + if (cert->sign_key) + cert->sign_len = len; +} + +char *tls_certificate_to_string(struct TLSCertificate *cert, char *buffer, int len) { + unsigned int i; + if (!buffer) + return NULL; + buffer[0] = 0; + if (cert->version) { + int res = snprintf(buffer, len, "X.509v%i certificate\n Issued by: [%s]%s (%s)\n Issued to: [%s]%s (%s, %s)\n Subject: %s\n Validity: %s - %s\n OCSP: %s\n Serial number: ", + (int)cert->version, + cert->issuer_country, cert->issuer_entity, cert->issuer_subject, + cert->country, cert->entity, cert->state, cert->location, + cert->subject, + cert->not_before, cert->not_after, + cert->ocsp + ); + if (res > 0) { + for (i = 0; i < cert->serial_len; i++) + res += snprintf(buffer + res, len - res, "%02x", (int)cert->serial_number[i]); + } + if ((cert->san) && (cert->san_length)) { + res += snprintf(buffer + res, len - res, "\n Alternative subjects: "); + for (i = 0; i < cert->san_length; i++) { + if (i) + res += snprintf(buffer + res, len - res, ", %s", cert->san[i]); + else + res += snprintf(buffer + res, len - res, "%s", cert->san[i]); + } + } + res += snprintf(buffer + res, len - res, "\n Key (%i bits, ", cert->pk_len * 8); + if (res > 0) { + switch (cert->key_algorithm) { + case TLS_RSA_SIGN_RSA: + res += snprintf(buffer + res, len - res, "RSA_SIGN_RSA"); + break; + case TLS_RSA_SIGN_MD5: + res += snprintf(buffer + res, len - res, "RSA_SIGN_MD5"); + break; + case TLS_RSA_SIGN_SHA1: + res += snprintf(buffer + res, len - res, "RSA_SIGN_SHA1"); + break; + case TLS_RSA_SIGN_SHA256: + res += snprintf(buffer + res, len - res, "RSA_SIGN_SHA256"); + break; + case TLS_RSA_SIGN_SHA384: + res += snprintf(buffer + res, len - res, "RSA_SIGN_SHA384"); + break; + case TLS_RSA_SIGN_SHA512: + res += snprintf(buffer + res, len - res, "RSA_SIGN_SHA512"); + break; + case TLS_ECDSA_SIGN_SHA256: + res += snprintf(buffer + res, len - res, "ECDSA_SIGN_SHA512"); + break; + case TLS_EC_PUBLIC_KEY: + res += snprintf(buffer + res, len - res, "EC_PUBLIC_KEY"); + break; + default: + res += snprintf(buffer + res, len - res, "not supported (%i)", (int)cert->key_algorithm); + } + } + if ((res > 0) && (cert->ec_algorithm)) { + switch (cert->ec_algorithm) { + case TLS_EC_prime192v1: + res += snprintf(buffer + res, len - res, " prime192v1"); + break; + case TLS_EC_prime192v2: + res += snprintf(buffer + res, len - res, " prime192v2"); + break; + case TLS_EC_prime192v3: + res += snprintf(buffer + res, len - res, " prime192v3"); + break; + case TLS_EC_prime239v2: + res += snprintf(buffer + res, len - res, " prime239v2"); + break; + case TLS_EC_secp256r1: + res += snprintf(buffer + res, len - res, " EC_secp256r1"); + break; + case TLS_EC_secp224r1: + res += snprintf(buffer + res, len - res, " EC_secp224r1"); + break; + case TLS_EC_secp384r1: + res += snprintf(buffer + res, len - res, " EC_secp384r1"); + break; + case TLS_EC_secp521r1: + res += snprintf(buffer + res, len - res, " EC_secp521r1"); + break; + default: + res += snprintf(buffer + res, len - res, " unknown(%i)", (int)cert->ec_algorithm); + } + } + res += snprintf(buffer + res, len - res, "):\n"); + if (res > 0) { + for (i = 0; i < cert->pk_len; i++) + res += snprintf(buffer + res, len - res, "%02x", (int)cert->pk[i]); + res += snprintf(buffer + res, len - res, "\n Signature (%i bits, ", cert->sign_len * 8); + switch (cert->algorithm) { + case TLS_RSA_SIGN_RSA: + res += snprintf(buffer + res, len - res, "RSA_SIGN_RSA):\n"); + break; + case TLS_RSA_SIGN_MD5: + res += snprintf(buffer + res, len - res, "RSA_SIGN_MD5):\n"); + break; + case TLS_RSA_SIGN_SHA1: + res += snprintf(buffer + res, len - res, "RSA_SIGN_SHA1):\n"); + break; + case TLS_RSA_SIGN_SHA256: + res += snprintf(buffer + res, len - res, "RSA_SIGN_SHA256):\n"); + break; + case TLS_RSA_SIGN_SHA384: + res += snprintf(buffer + res, len - res, "RSA_SIGN_SHA384):\n"); + break; + case TLS_RSA_SIGN_SHA512: + res += snprintf(buffer + res, len - res, "RSA_SIGN_SHA512):\n"); + break; + case TLS_EC_PUBLIC_KEY: + res += snprintf(buffer + res, len - res, "EC_PUBLIC_KEY):\n"); + break; + default: + res += snprintf(buffer + res, len - res, "not supported):\n"); + } + + for (i = 0; i < cert->sign_len; i++) + res += snprintf(buffer + res, len - res, "%02x", (int)cert->sign_key[i]); + } + } else + if ((cert->priv) && (cert->priv_len)) { + int res = snprintf(buffer, len, "X.509 private key\n"); + res += snprintf(buffer + res, len - res, " Private Key: "); + if (res > 0) { + for (i = 0; i < cert->priv_len; i++) + res += snprintf(buffer + res, len - res, "%02x", (int)cert->priv[i]); + } + } else + snprintf(buffer, len, "Empty ASN1 file"); + return buffer; +} + +void tls_certificate_set_exponent(struct TLSCertificate *cert, const unsigned char *val, int len) { + tls_certificate_set_copy(&cert->exponent, val, len); + if (cert->exponent) + cert->exponent_len = len; +} + +void tls_certificate_set_serial(struct TLSCertificate *cert, const unsigned char *val, int len) { + tls_certificate_set_copy(&cert->serial_number, val, len); + if (cert->serial_number) + cert->serial_len = len; +} + +void tls_certificate_set_algorithm(struct TLSContext *context, unsigned int *algorithm, const unsigned char *val, int len) { + if ((len == 7) && (_is_oid(val, TLS_EC_PUBLIC_KEY_OID, 7))) { + *algorithm = TLS_EC_PUBLIC_KEY; + return; + } + if (len == 8) { + if (_is_oid(val, TLS_EC_prime192v1_OID, len)) { + *algorithm = TLS_EC_prime192v1; + return; + } + if (_is_oid(val, TLS_EC_prime192v2_OID, len)) { + *algorithm = TLS_EC_prime192v2; + return; + } + if (_is_oid(val, TLS_EC_prime192v3_OID, len)) { + *algorithm = TLS_EC_prime192v3; + return; + } + if (_is_oid(val, TLS_EC_prime239v1_OID, len)) { + *algorithm = TLS_EC_prime239v1; + return; + } + if (_is_oid(val, TLS_EC_prime239v2_OID, len)) { + *algorithm = TLS_EC_prime239v2; + return; + } + if (_is_oid(val, TLS_EC_prime239v3_OID, len)) { + *algorithm = TLS_EC_prime239v3; + return; + } + if (_is_oid(val, TLS_EC_prime256v1_OID, len)) { + *algorithm = TLS_EC_prime256v1; + return; + } + } + if (len == 5) { + if (_is_oid2(val, TLS_EC_secp224r1_OID, len, sizeof(TLS_EC_secp224r1_OID) - 1)) { + *algorithm = TLS_EC_secp224r1; + return; + } + if (_is_oid2(val, TLS_EC_secp384r1_OID, len, sizeof(TLS_EC_secp384r1_OID) - 1)) { + *algorithm = TLS_EC_secp384r1; + return; + } + if (_is_oid2(val, TLS_EC_secp521r1_OID, len, sizeof(TLS_EC_secp521r1_OID) - 1)) { + *algorithm = TLS_EC_secp521r1; + return; + } + } + if (len != 9) + return; + + if (_is_oid(val, TLS_RSA_SIGN_SHA256_OID, 9)) { + *algorithm = TLS_RSA_SIGN_SHA256; + return; + } + + if (_is_oid(val, TLS_RSA_SIGN_RSA_OID, 9)) { + *algorithm = TLS_RSA_SIGN_RSA; + return; + } + + if (_is_oid(val, TLS_RSA_SIGN_SHA1_OID, 9)) { + *algorithm = TLS_RSA_SIGN_SHA1; + return; + } + + if (_is_oid(val, TLS_RSA_SIGN_SHA512_OID, 9)) { + *algorithm = TLS_RSA_SIGN_SHA512; + return; + } + + if (_is_oid(val, TLS_RSA_SIGN_SHA384_OID, 9)) { + *algorithm = TLS_RSA_SIGN_SHA384; + return; + } + + if (_is_oid(val, TLS_RSA_SIGN_MD5_OID, 9)) { + *algorithm = TLS_RSA_SIGN_MD5; + return; + } + + if (_is_oid(val, TLS_ECDSA_SIGN_SHA256_OID, 9)) { + *algorithm = TLS_ECDSA_SIGN_SHA256; + return; + } + // client should fail on unsupported signature + if (!context->is_server) { + DEBUG_PRINT("UNSUPPORTED SIGNATURE ALGORITHM\n"); + context->critical_error = 1; + } +} + +void tls_destroy_certificate(struct TLSCertificate *cert) { + if (cert) { + int i; + TLS_FREE(cert->exponent); + TLS_FREE(cert->pk); + TLS_FREE(cert->issuer_country); + TLS_FREE(cert->issuer_state); + TLS_FREE(cert->issuer_location); + TLS_FREE(cert->issuer_entity); + TLS_FREE(cert->issuer_subject); + TLS_FREE(cert->country); + TLS_FREE(cert->state); + TLS_FREE(cert->location); + TLS_FREE(cert->subject); + for (i = 0; i < cert->san_length; i++) { + TLS_FREE(cert->san[i]); + } + TLS_FREE(cert->san); + TLS_FREE(cert->ocsp); + TLS_FREE(cert->serial_number); + TLS_FREE(cert->entity); + TLS_FREE(cert->not_before); + TLS_FREE(cert->not_after); + TLS_FREE(cert->sign_key); + TLS_FREE(cert->priv); + TLS_FREE(cert->der_bytes); + TLS_FREE(cert->bytes); + TLS_FREE(cert->fingerprint); + TLS_FREE(cert); + } +} + +struct TLSPacket *tls_create_packet(struct TLSContext *context, unsigned char type, unsigned short version, int payload_size_hint) { + struct TLSPacket *packet = (struct TLSPacket *)TLS_MALLOC(sizeof(struct TLSPacket)); + if (!packet) + return NULL; + packet->broken = 0; + if (payload_size_hint > 0) + packet->size = payload_size_hint + 10; + else + packet->size = TLS_BLOB_INCREMENT; + packet->buf = (unsigned char *)TLS_MALLOC(packet->size); + packet->context = context; + if (!packet->buf) { + TLS_FREE(packet); + return NULL; + } + if ((context) && (context->dtls)) + packet->len = 13; + else + packet->len = 5; + packet->buf[0] = type; +#ifdef WITH_TLS_13 + switch (version) { + case TLS_V13: + // check if context is not null. If null, is a tls_export_context call + if (context) + *(unsigned short *)(packet->buf + 1) = 0x0303; // no need to reorder (same bytes) + else + *(unsigned short *)(packet->buf + 1) = htons(version); + break; + case DTLS_V13: + *(unsigned short *)(packet->buf + 1) = htons(DTLS_V13); + break; + default: + *(unsigned short *)(packet->buf + 1) = htons(version); + } +#else + *(unsigned short *)(packet->buf + 1) = htons(version); +#endif + return packet; +} + +void tls_destroy_packet(struct TLSPacket *packet) { + if (packet) { + if (packet->buf) + TLS_FREE(packet->buf); + TLS_FREE(packet); + } +} + +int _private_tls_crypto_create(struct TLSContext *context, int key_length, unsigned char *localkey, unsigned char *localiv, unsigned char *remotekey, unsigned char *remoteiv) { + if (context->crypto.created) { + if (context->crypto.created == 1) { + cbc_done(&context->crypto.ctx_remote.aes_remote); + cbc_done(&context->crypto.ctx_local.aes_local); + } else { +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (context->crypto.created == 2) { +#endif + unsigned char dummy_buffer[32]; + unsigned long tag_len = 0; + gcm_done(&context->crypto.ctx_remote.aes_gcm_remote, dummy_buffer, &tag_len); + gcm_done(&context->crypto.ctx_local.aes_gcm_local, dummy_buffer, &tag_len); +#ifdef TLS_WITH_CHACHA20_POLY1305 + } +#endif + } + context->crypto.created = 0; + } + tls_init(); + int is_aead = _private_tls_is_aead(context); + int cipherID = find_cipher("aes"); + DEBUG_PRINT("Using cipher ID: %x\n", (int)context->cipher); +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) { + unsigned int counter = 1; + + chacha_keysetup(&context->crypto.ctx_local.chacha_local, localkey, key_length * 8); + chacha_ivsetup_96bitnonce(&context->crypto.ctx_local.chacha_local, localiv, (unsigned char *)&counter); + + chacha_keysetup(&context->crypto.ctx_remote.chacha_remote, remotekey, key_length * 8); + chacha_ivsetup_96bitnonce(&context->crypto.ctx_remote.chacha_remote, remoteiv, (unsigned char *)&counter); + + context->crypto.created = 3; + } else +#endif + if (is_aead) { + int res1 = gcm_init(&context->crypto.ctx_local.aes_gcm_local, cipherID, localkey, key_length); + int res2 = gcm_init(&context->crypto.ctx_remote.aes_gcm_remote, cipherID, remotekey, key_length); + + if ((res1) || (res2)) + return TLS_GENERIC_ERROR; + context->crypto.created = 2; + } else { + int res1 = cbc_start(cipherID, localiv, localkey, key_length, 0, &context->crypto.ctx_local.aes_local); + int res2 = cbc_start(cipherID, remoteiv, remotekey, key_length, 0, &context->crypto.ctx_remote.aes_remote); + + if ((res1) || (res2)) + return TLS_GENERIC_ERROR; + context->crypto.created = 1; + } + return 0; +} + +int _private_tls_crypto_encrypt(struct TLSContext *context, unsigned char *buf, unsigned char *ct, unsigned int len) { + if (context->crypto.created == 1) + return cbc_encrypt(buf, ct, len, &context->crypto.ctx_local.aes_local); + + memset(ct, 0, len); + return TLS_GENERIC_ERROR; +} + +int _private_tls_crypto_decrypt(struct TLSContext *context, unsigned char *buf, unsigned char *pt, unsigned int len) { + if (context->crypto.created == 1) + return cbc_decrypt(buf, pt, len, &context->crypto.ctx_remote.aes_remote); + + memset(pt, 0, len); + return TLS_GENERIC_ERROR; +} + +void _private_tls_crypto_done(struct TLSContext *context) { + unsigned char dummy_buffer[32]; + unsigned long tag_len = 0; + switch (context->crypto.created) { + case 1: + cbc_done(&context->crypto.ctx_remote.aes_remote); + cbc_done(&context->crypto.ctx_local.aes_local); + break; + case 2: + gcm_done(&context->crypto.ctx_remote.aes_gcm_remote, dummy_buffer, &tag_len); + gcm_done(&context->crypto.ctx_local.aes_gcm_local, dummy_buffer, &tag_len); + break; + } + context->crypto.created = 0; +} + +void tls_packet_update(struct TLSPacket *packet) { + if ((packet) && (!packet->broken)) { + int footer_size = 0; +#ifdef WITH_TLS_13 + if ((packet->context) && ((packet->context->version == TLS_V13) || (packet->context->version == DTLS_V13)) && (packet->context->cipher_spec_set) && (packet->context->crypto.created)) { + // type + tls_packet_uint8(packet, packet->buf[0]); + // no padding + // tls_packet_uint8(packet, 0); + footer_size = 1; + } +#endif + unsigned int header_size = 5; + if ((packet->context) && (packet->context->dtls)) { + header_size = 13; + *(unsigned short *)(packet->buf + 3) = htons(packet->context->dtls_epoch_local); + uint64_t sequence_number = packet->context->local_sequence_number; + packet->buf[5] = (unsigned char)(sequence_number / 0x10000000000LL); + sequence_number %= 0x10000000000LL; + packet->buf[6] = (unsigned char)(sequence_number / 0x100000000LL); + sequence_number %= 0x100000000LL; + packet->buf[7] = (unsigned char)(sequence_number / 0x1000000); + sequence_number %= 0x1000000; + packet->buf[8] = (unsigned char)(sequence_number / 0x10000); + sequence_number %= 0x10000; + packet->buf[9] = (unsigned char)(sequence_number / 0x100); + sequence_number %= 0x100; + packet->buf[10] = (unsigned char)sequence_number; + + *(unsigned short *)(packet->buf + 11) = htons(packet->len - header_size); + } else + *(unsigned short *)(packet->buf + 3) = htons(packet->len - header_size); + if (packet->context) { + if (packet->buf[0] != TLS_CHANGE_CIPHER) { + if ((packet->buf[0] == TLS_HANDSHAKE) && (packet->len > header_size)) { + unsigned char handshake_type = packet->buf[header_size]; + if ((handshake_type != 0x00) && (handshake_type != 0x03)) + _private_tls_update_hash(packet->context, packet->buf + header_size, packet->len - header_size - footer_size, 1, 0); + } +#ifdef TLS_12_FALSE_START + if (((packet->context->cipher_spec_set) || (packet->context->false_start)) && (packet->context->crypto.created)) { +#else + if ((packet->context->cipher_spec_set) && (packet->context->crypto.created)) { +#endif + int block_size = TLS_AES_BLOCK_SIZE; + int mac_size = 0; + unsigned int length = 0; + unsigned char padding = 0; + unsigned int pt_length = packet->len - header_size; + + if (packet->context->crypto.created == 1) { + mac_size = _private_tls_mac_length(packet->context); +#ifdef TLS_LEGACY_SUPPORT + if (packet->context->version == TLS_V10) + length = packet->len - header_size + mac_size; + else +#endif + length = packet->len - header_size + TLS_AES_IV_LENGTH + mac_size; + padding = block_size - length % block_size; + length += padding; +#ifdef TLS_WITH_CHACHA20_POLY1305 + } else + if (packet->context->crypto.created == 3) { + mac_size = POLY1305_TAGLEN; + length = packet->len - header_size + mac_size; +#endif + } else { + mac_size = TLS_GCM_TAG_LEN; + length = packet->len - header_size + 8 + mac_size; + } + if (packet->context->crypto.created == 1) { + unsigned char *buf = (unsigned char *)TLS_MALLOC(length); + if (buf) { + unsigned char *ct = (unsigned char *)TLS_MALLOC(length + header_size); + if (ct) { + unsigned int buf_pos = 0; + memcpy(ct, packet->buf, header_size - 2); + *(unsigned short *)&ct[header_size - 2] = htons(length); +#ifdef TLS_LEGACY_SUPPORT + if (packet->context->version != TLS_V10) +#endif + { + tls_random(buf, TLS_AES_IV_LENGTH); + buf_pos += TLS_AES_IV_LENGTH; + } + // copy payload + memcpy(buf + buf_pos, packet->buf + header_size, packet->len - header_size); + buf_pos += packet->len - header_size; + if (packet->context->dtls) { + unsigned char temp_buf[5]; + memcpy(temp_buf, packet->buf, 3); + *(unsigned short *)(temp_buf + 3) = *(unsigned short *)&packet->buf[header_size - 2]; + uint64_t dtls_sequence_number = ntohll(*(uint64_t *)&packet->buf[3]); + _private_tls_hmac_message(1, packet->context, temp_buf, 5, packet->buf + header_size, packet->len - header_size, buf + buf_pos, mac_size, dtls_sequence_number); + } else + _private_tls_hmac_message(1, packet->context, packet->buf, packet->len, NULL, 0, buf + buf_pos, mac_size, 0); + buf_pos += mac_size; + + memset(buf + buf_pos, padding - 1, padding); + buf_pos += padding; + + //DEBUG_DUMP_HEX_LABEL("PT BUFFER", buf, length); + _private_tls_crypto_encrypt(packet->context, buf, ct + header_size, length); + TLS_FREE(packet->buf); + packet->buf = ct; + packet->len = length + header_size; + packet->size = packet->len; + } else { + // invalidate packet + memset(packet->buf, 0, packet->len); + } + TLS_FREE(buf); + } else { + // invalidate packet + memset(packet->buf, 0, packet->len); + } + } else +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (packet->context->crypto.created >= 2) { +#else + if (packet->context->crypto.created == 2) { +#endif + // + 1 = type + int ct_size = length + header_size + 12 + TLS_MAX_TAG_LEN + 1; + unsigned char *ct = (unsigned char *)TLS_MALLOC(ct_size); + if (ct) { + memset(ct, 0, ct_size); + // AEAD + // sequence number (8 bytes) + // content type (1 byte) + // version (2 bytes) + // length (2 bytes) + unsigned char aad[13]; + int aad_size = sizeof(aad); + unsigned char *sequence = aad; +#ifdef WITH_TLS_13 + if ((packet->context->version == TLS_V13) || (packet->context->version == DTLS_V13)) { + aad[0] = TLS_APPLICATION_DATA; + aad[1] = packet->buf[1]; + aad[2] = packet->buf[2]; +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (packet->context->crypto.created == 3) + *((unsigned short *)(aad + 3)) = htons(packet->len + POLY1305_TAGLEN - header_size); + else +#endif + *((unsigned short *)(aad + 3)) = htons(packet->len + TLS_GCM_TAG_LEN - header_size); + aad_size = 5; + sequence = aad + 5; + if (packet->context->dtls) + *((uint64_t *)sequence) = *(uint64_t *)&packet->buf[3]; + else + *((uint64_t *)sequence) = htonll(packet->context->local_sequence_number); + } else { +#endif + if (packet->context->dtls) + *((uint64_t *)aad) = *(uint64_t *)&packet->buf[3]; + else + *((uint64_t *)aad) = htonll(packet->context->local_sequence_number); + aad[8] = packet->buf[0]; + aad[9] = packet->buf[1]; + aad[10] = packet->buf[2]; + *((unsigned short *)(aad + 11)) = htons(packet->len - header_size); +#ifdef WITH_TLS_13 + } +#endif + int ct_pos = header_size; +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (packet->context->crypto.created == 3) { + unsigned int counter = 1; + unsigned char poly1305_key[POLY1305_KEYLEN]; + chacha_ivupdate(&packet->context->crypto.ctx_local.chacha_local, packet->context->crypto.ctx_local_mac.local_aead_iv, sequence, (u8 *)&counter); + chacha20_poly1305_key(&packet->context->crypto.ctx_local.chacha_local, poly1305_key); + ct_pos += chacha20_poly1305_aead(&packet->context->crypto.ctx_local.chacha_local, packet->buf + header_size, pt_length, aad, aad_size, poly1305_key, ct + ct_pos); + } else { +#endif + unsigned char iv[TLS_13_AES_GCM_IV_LENGTH]; +#ifdef WITH_TLS_13 + if ((packet->context->version == TLS_V13) || (packet->context->version == DTLS_V13)) { + memcpy(iv, packet->context->crypto.ctx_local_mac.local_iv, TLS_13_AES_GCM_IV_LENGTH); + int i; + int offset = TLS_13_AES_GCM_IV_LENGTH - 8; + for (i = 0; i < 8; i++) + iv[offset + i] = packet->context->crypto.ctx_local_mac.local_iv[offset + i] ^ sequence[i]; + } else { +#endif + memcpy(iv, packet->context->crypto.ctx_local_mac.local_aead_iv, TLS_AES_GCM_IV_LENGTH); + tls_random(iv + TLS_AES_GCM_IV_LENGTH, 8); + memcpy(ct + ct_pos, iv + TLS_AES_GCM_IV_LENGTH, 8); + ct_pos += 8; +#ifdef WITH_TLS_13 + } +#endif + + gcm_reset(&packet->context->crypto.ctx_local.aes_gcm_local); + gcm_add_iv(&packet->context->crypto.ctx_local.aes_gcm_local, iv, 12); + gcm_add_aad(&packet->context->crypto.ctx_local.aes_gcm_local, aad, aad_size); + gcm_process(&packet->context->crypto.ctx_local.aes_gcm_local, packet->buf + header_size, pt_length, ct + ct_pos, GCM_ENCRYPT); + ct_pos += pt_length; + + unsigned long taglen = TLS_GCM_TAG_LEN; + gcm_done(&packet->context->crypto.ctx_local.aes_gcm_local, ct + ct_pos, &taglen); + ct_pos += taglen; +#ifdef TLS_WITH_CHACHA20_POLY1305 + } +#endif +#ifdef WITH_TLS_13 + if ((packet->context->version == TLS_V13) || (packet->context->version == DTLS_V13)) { + ct[0] = TLS_APPLICATION_DATA; + *(unsigned short *)&ct[1] = htons(packet->context->version == TLS_V13 ? TLS_V12 : DTLS_V12); + // is dtls ? + if (header_size != 5) + memcpy(ct, packet->buf + 3, header_size - 2); + } else +#endif + memcpy(ct, packet->buf, header_size - 2); + *(unsigned short *)&ct[header_size - 2] = htons(ct_pos - header_size); + TLS_FREE(packet->buf); + packet->buf = ct; + packet->len = ct_pos; + packet->size = ct_pos; + } else { + // invalidate packet + memset(packet->buf, 0, packet->len); + } + } else { + // invalidate packet (never reached) + memset(packet->buf, 0, packet->len); + } + } + } else + packet->context->dtls_epoch_local++; + packet->context->local_sequence_number++; + } + } +} + +int tls_packet_append(struct TLSPacket *packet, const unsigned char *buf, unsigned int len) { + if ((!packet) || (packet->broken)) + return -1; + + if (!len) + return 0; + + unsigned int new_len = packet->len + len; + + if (new_len > packet->size) { + packet->size = (new_len / TLS_BLOB_INCREMENT + 1) * TLS_BLOB_INCREMENT; + packet->buf = (unsigned char *)TLS_REALLOC(packet->buf, packet->size); + if (!packet->buf) { + packet->size = 0; + packet->len = 0; + packet->broken = 1; + return -1; + } + } + memcpy(packet->buf + packet->len, buf, len); + packet->len = new_len; + return new_len; +} + +int tls_packet_uint8(struct TLSPacket *packet, unsigned char i) { + return tls_packet_append(packet, &i, 1); +} + +int tls_packet_uint16(struct TLSPacket *packet, unsigned short i) { + unsigned short ni = htons(i); + return tls_packet_append(packet, (unsigned char *)&ni, 2); +} + +int tls_packet_uint32(struct TLSPacket *packet, unsigned int i) { + unsigned int ni = htonl(i); + return tls_packet_append(packet, (unsigned char *)&ni, 4); +} + +int tls_packet_uint24(struct TLSPacket *packet, unsigned int i) { + unsigned char buf[3]; + buf[0] = i / 0x10000; + i %= 0x10000; + buf[1] = i / 0x100; + i %= 0x100; + buf[2] = i; + + return tls_packet_append(packet, buf, 3); +} + +int tls_random(unsigned char *key, int len) { + for (int i = 0; i < len; i++) + key[i] = rand() & 0xff; + return 1; +} + +TLSHash *_private_tls_ensure_hash(struct TLSContext *context) { + TLSHash *hash = context->handshake_hash; + if (!hash) { + hash = (TLSHash *)TLS_MALLOC(sizeof(TLSHash)); + if (hash) + memset(hash, 0, sizeof(TLSHash)); + context->handshake_hash = hash; + } + return hash; +} + +void _private_tls_destroy_hash(struct TLSContext *context) { + if (context) { + TLS_FREE(context->handshake_hash); + context->handshake_hash = NULL; + } +} + +void _private_tls_create_hash(struct TLSContext *context) { + if (!context) + return; + TLSHash *hash = _private_tls_ensure_hash(context); + if ((context->version == TLS_V12) || (context->version == DTLS_V12) || (context->version == TLS_V13) || (context->version == DTLS_V13)) { + int hash_size = _private_tls_mac_length(context); + if (hash->created) { + unsigned char temp[TLS_MAX_SHA_SIZE]; + sha384_done(&hash->hash32, temp); + sha256_done(&hash->hash48, temp); + } + sha384_init(&hash->hash48); + sha256_init(&hash->hash32); + hash->created = 1; + } else { +#ifdef TLS_LEGACY_SUPPORT + // TLS_V11 + if (hash->created) { + unsigned char temp[TLS_V11_HASH_SIZE]; + md5_done(&hash->hash32, temp); + sha1_done(&hash->hash2, temp); + } + md5_init(&hash->hash32); + sha1_init(&hash->hash2); + hash->created = 1; +#endif + } +} + +void _private_tls_update_handshake_list(struct TLSContext *context, const unsigned char *in, unsigned int len, unsigned char direction, unsigned char connection_status) { + if ((!context) || (!context->dtls) || (!in) || (!len)) + return; + + struct TLSHandshakeList *msg = (struct TLSHandshakeList *)TLS_MALLOC(sizeof(struct TLSHandshakeList)); + if (!msg) + return; + + msg->msg = (unsigned char *)TLS_MALLOC(len); + if (!msg) { + TLS_FREE(msg); + return; + } + + memcpy(msg->msg, in, len); + msg->len = len; + msg->direction = direction; + msg->connection_status = connection_status ? connection_status : context->connection_status; + msg->next = NULL; + + if (!context->dtls_data->dtls_handshake_list) { + context->dtls_data->dtls_handshake_list = msg; + return; + } + + struct TLSHandshakeList *last = context->dtls_data->dtls_handshake_list; + while (last->next) + last = (struct TLSHandshakeList *)last->next; + + last->next = msg; +} + +int _private_tls_update_hash(struct TLSContext *context, const unsigned char *in, unsigned int len, unsigned char direction, unsigned char connection_status) { + if (!context) + return 0; + + if (context->dtls) + _private_tls_update_handshake_list(context, in, len, direction, connection_status); + + TLSHash *hash = _private_tls_ensure_hash(context); + if ((context->version == TLS_V12) || (context->version == DTLS_V12) || (context->version == TLS_V13) || (context->version == DTLS_V13)) { + if (!hash->created) { + _private_tls_create_hash(context); +#ifdef TLS_LEGACY_SUPPORT + // cache first hello in case of protocol downgrade + if ((!context->is_server) && (!context->cached_handshake) && (!context->request_client_certificate) && (len)) { + context->cached_handshake = (unsigned char *)TLS_MALLOC(len); + if (context->cached_handshake) { + memcpy(context->cached_handshake, in, len); + context->cached_handshake_len = len; + } + } +#endif + } + int hash_size = _private_tls_mac_length(context); + sha256_process(&hash->hash32, in, len); + sha384_process(&hash->hash48, in, len); + if (!hash_size) + hash_size = TLS_SHA256_MAC_SIZE; + } else { +#ifdef TLS_LEGACY_SUPPORT + if (!hash->created) + _private_tls_create_hash(context); + md5_process(&hash->hash32, in, len); + sha1_process(&hash->hash2, in, len); +#endif + } + if ((context->request_client_certificate) && (len)) { + // cache all messages for verification + int new_len = context->cached_handshake_len + len; + context->cached_handshake = (unsigned char *)TLS_REALLOC(context->cached_handshake, new_len); + if (context->cached_handshake) { + memcpy(context->cached_handshake + context->cached_handshake_len, in, len); + context->cached_handshake_len = new_len; + } else + context->cached_handshake_len = 0; + } + return 0; +} + +#ifdef TLS_LEGACY_SUPPORT +int _private_tls_change_hash_type(struct TLSContext *context) { + if (!context) + return 0; + TLSHash *hash = _private_tls_ensure_hash(context); + if ((hash) && (hash->created) && (context->cached_handshake) && (context->cached_handshake_len)) { + _private_tls_destroy_hash(context); + int res = _private_tls_update_hash(context, context->cached_handshake, context->cached_handshake_len, 0, 0); + TLS_FREE(context->cached_handshake); + context->cached_handshake = NULL; + context->cached_handshake_len = 0; + return res; + } + return 0; +} +#endif + +int _private_tls_done_hash(struct TLSContext *context, unsigned char *hout) { + if (!context) + return 0; + + TLSHash *hash = _private_tls_ensure_hash(context); + if (!hash->created) + return 0; + + int hash_size = 0; + if ((context->version == TLS_V12) || (context->version == DTLS_V12) || (context->version == TLS_V13) || (context->version == DTLS_V13)) { + unsigned char temp[TLS_MAX_SHA_SIZE]; + if (!hout) + hout = temp; + //TLS_HASH_DONE(&hash->hash, hout); + hash_size = _private_tls_mac_length(context); + if (hash_size == TLS_SHA384_MAC_SIZE) { + sha256_done(&hash->hash32, temp); + sha384_done(&hash->hash48, hout); + } else { + sha256_done(&hash->hash32, hout); + sha384_done(&hash->hash48, temp); + hash_size = TLS_SHA256_MAC_SIZE; + } + } else { +#ifdef TLS_LEGACY_SUPPORT + // TLS_V11 + unsigned char temp[TLS_V11_HASH_SIZE]; + if (!hout) + hout = temp; + md5_done(&hash->hash32, hout); + sha1_done(&hash->hash2, hout + 16); + hash_size = TLS_V11_HASH_SIZE; +#endif + } + hash->created = 0; + if (context->cached_handshake) { + // not needed anymore + TLS_FREE(context->cached_handshake); + context->cached_handshake = NULL; + context->cached_handshake_len = 0; + } + return hash_size; +} + +int _private_tls_get_hash_idx(struct TLSContext *context) { + if (!context) + return -1; + switch (_private_tls_mac_length(context)) { + case TLS_SHA256_MAC_SIZE: + return find_hash("sha256"); + case TLS_SHA384_MAC_SIZE: + return find_hash("sha384"); + case TLS_SHA1_MAC_SIZE: + return find_hash("sha1"); + } + return -1; +} + +int _private_tls_get_hash(struct TLSContext *context, unsigned char *hout) { + if (!context) + return 0; + + TLSHash *hash = _private_tls_ensure_hash(context); + if (!hash->created) + return 0; + + int hash_size = 0; + if ((context->version == TLS_V12) || (context->version == DTLS_V12) || (context->version == TLS_V13) || (context->version == DTLS_V13)) { + hash_size = _private_tls_mac_length(context); + hash_state prec; + if (hash_size == TLS_SHA384_MAC_SIZE) { + memcpy(&prec, &hash->hash48, sizeof(hash_state)); + sha384_done(&hash->hash48, hout); + memcpy(&hash->hash48, &prec, sizeof(hash_state)); + } else { + memcpy(&prec, &hash->hash32, sizeof(hash_state)); + hash_size = TLS_SHA256_MAC_SIZE; + sha256_done(&hash->hash32, hout); + memcpy(&hash->hash32, &prec, sizeof(hash_state)); + } + } else { +#ifdef TLS_LEGACY_SUPPORT + // TLS_V11 + hash_state prec; + + memcpy(&prec, &hash->hash32, sizeof(hash_state)); + md5_done(&hash->hash32, hout); + memcpy(&hash->hash32, &prec, sizeof(hash_state)); + + memcpy(&prec, &hash->hash2, sizeof(hash_state)); + sha1_done(&hash->hash2, hout + 16); + memcpy(&hash->hash2, &prec, sizeof(hash_state)); + + hash_size = TLS_V11_HASH_SIZE; +#endif + } + return hash_size; +} + +int _private_tls_write_packet(struct TLSPacket *packet) { + if (!packet) + return -1; + struct TLSContext *context = packet->context; + if (!context) + return -1; + + if (context->tls_buffer) { + int len = context->tls_buffer_len + packet->len; + context->tls_buffer = (unsigned char *)TLS_REALLOC(context->tls_buffer, len); + if (!context->tls_buffer) { + context->tls_buffer_len = 0; + return -1; + } + memcpy(context->tls_buffer + context->tls_buffer_len, packet->buf, packet->len); + context->tls_buffer_len = len; + int written = packet->len; + tls_destroy_packet(packet); + return written; + } + context->tls_buffer_len = packet->len; + context->tls_buffer = packet->buf; + packet->buf = NULL; + packet->len = 0; + packet->size = 0; + tls_destroy_packet(packet); + return context->tls_buffer_len; +} + +int _private_tls_write_app_data(struct TLSContext *context, const unsigned char *buf, unsigned int buf_len) { + if (!context) + return -1; + if ((!buf) || (!buf_len)) + return 0; + + int len = context->application_buffer_len + buf_len; + context->application_buffer = (unsigned char *)TLS_REALLOC(context->application_buffer, len); + if (!context->application_buffer) { + context->application_buffer_len = 0; + return -1; + } + memcpy(context->application_buffer + context->application_buffer_len, buf, buf_len); + context->application_buffer_len = len; + return buf_len; +} + +const unsigned char *tls_get_write_buffer(struct TLSContext *context, unsigned int *outlen) { + if (!outlen) + return NULL; + if (!context) { + *outlen = 0; + return NULL; + } + // check if any error + if (context->sleep_until) { + if (context->sleep_until < time(NULL)) { + *outlen = 0; + return NULL; + } + context->sleep_until = 0; + } + *outlen = context->tls_buffer_len; + return context->tls_buffer; +} + +const unsigned char *tls_get_message(struct TLSContext *context, unsigned int *outlen, unsigned int offset) { + if (!outlen) + return NULL; + if ((!context) || (!context->tls_buffer)) { + *outlen = 0; + return NULL; + } + + if (offset >= context->tls_buffer_len) { + *outlen = 0; + return NULL; + } + // check if any error + if (context->sleep_until) { + if (context->sleep_until < time(NULL)) { + *outlen = 0; + return NULL; + } + context->sleep_until = 0; + } + unsigned char *tls_buffer = &context->tls_buffer[offset]; + unsigned int tls_buffer_len = context->tls_buffer_len - offset; + unsigned int len = 0; + if (context->dtls) { + if (tls_buffer_len < 13) { + *outlen = 0; + return NULL; + } + + len = ntohs(*(unsigned short *)&tls_buffer[11]) + 13; + } else { + if (tls_buffer_len < 5) { + *outlen = 0; + return NULL; + } + len = ntohs(*(unsigned short *)&tls_buffer[3]) + 5; + } + if (len > tls_buffer_len) { + *outlen = 0; + return NULL; + } + + *outlen = len; + return tls_buffer; +} + +void tls_buffer_clear(struct TLSContext *context) { + if ((context) && (context->tls_buffer)) { + TLS_FREE(context->tls_buffer); + context->tls_buffer = NULL; + context->tls_buffer_len = 0; + } +} + +int tls_established(struct TLSContext *context) { + if (context) { + if (context->critical_error) + return -1; + + if (context->connection_status == 0xFF) + return 1; + +#ifdef TLS_12_FALSE_START + // allow false start + if ((!context->is_server) && (context->version == TLS_V12) && (context->false_start)) + return 1; +#endif + } + return 0; +} + +void tls_read_clear(struct TLSContext *context) { + if ((context) && (context->application_buffer)) { + TLS_FREE(context->application_buffer); + context->application_buffer = NULL; + context->application_buffer_len = 0; + } +} + +int tls_read(struct TLSContext *context, unsigned char *buf, unsigned int size) { + if (!context) + return -1; + if ((context->application_buffer) && (context->application_buffer_len)) { + if (context->application_buffer_len < size) + size = context->application_buffer_len; + + memcpy(buf, context->application_buffer, size); + if (context->application_buffer_len == size) { + TLS_FREE(context->application_buffer); + context->application_buffer = NULL; + context->application_buffer_len = 0; + return size; + } + context->application_buffer_len -= size; + memmove(context->application_buffer, context->application_buffer + size, context->application_buffer_len); + return size; + } + return 0; +} + +struct TLSContext *tls_create_context(unsigned char is_server, unsigned short version) { + struct TLSContext *context = (struct TLSContext *)TLS_MALLOC(sizeof(struct TLSContext)); + if (context) { + memset(context, 0, sizeof(struct TLSContext)); + context->is_server = is_server; + if ((version == DTLS_V13) || (version == DTLS_V12) || (version == DTLS_V10)) { + context->dtls = 1; + context->dtls_data = (struct DTLSData *)TLS_MALLOC(sizeof(struct DTLSData)); + if (!context->dtls_data) { + TLS_FREE(context); + return NULL; + } + memset(context->dtls_data, 0, sizeof(struct DTLSData)); + } + context->version = version; + } + return context; +} + +#ifdef TLS_FORWARD_SECRECY +const struct ECCCurveParameters *tls_set_curve(struct TLSContext *context, const struct ECCCurveParameters *curve) { + if (!context->is_server) + return NULL; + const struct ECCCurveParameters *old_curve = context->curve; + context->curve = curve; + return old_curve; +} +#endif + +struct TLSContext *tls_accept(struct TLSContext *context) { + if ((!context) || (!context->is_server)) + return NULL; + + struct TLSContext *child = (struct TLSContext *)TLS_MALLOC(sizeof(struct TLSContext)); + if (child) { + memset(child, 0, sizeof(struct TLSContext)); + child->is_server = 1; + child->is_child = 1; + child->dtls = context->dtls; + if (context->dtls) { + child->dtls_data = (struct DTLSData *)TLS_MALLOC(sizeof(struct DTLSData)); + if (!child->dtls_data) { + TLS_FREE(child); + return NULL; + } + memset(child->dtls_data, 0, sizeof(struct DTLSData)); + } + child->version = context->version; + child->certificates = context->certificates; + child->certificates_count = context->certificates_count; + child->private_key = context->private_key; +#ifdef TLS_ECDSA_SUPPORTED + child->ec_private_key = context->ec_private_key; +#endif + child->exportable = context->exportable; + child->root_certificates = context->root_certificates; + child->root_count = context->root_count; +#ifdef TLS_FORWARD_SECRECY + child->default_dhe_p = context->default_dhe_p; + child->default_dhe_g = context->default_dhe_g; + child->curve = context->curve; +#endif + child->alpn = context->alpn; + child->alpn_count = context->alpn_count; + child->request_client_certificate = context->request_client_certificate; + } + return child; +} + +#ifdef TLS_FORWARD_SECRECY +void _private_tls_dhe_free(struct TLSContext *context) { + if (context->dhe) { + _private_tls_dh_clear_key(context->dhe); + TLS_FREE(context->dhe); + context->dhe = NULL; + } +} + +void _private_tls_dhe_create(struct TLSContext *context) { + _private_tls_dhe_free(context); + context->dhe = (DHKey *)TLS_MALLOC(sizeof(DHKey)); + if (context->dhe) + memset(context->dhe, 0, sizeof(DHKey)); +} + +void _private_tls_ecc_dhe_free(struct TLSContext *context) { + if (context->ecc_dhe) { + ecc_free(context->ecc_dhe); + TLS_FREE(context->ecc_dhe); + context->ecc_dhe = NULL; + } +} + +void _private_tls_ecc_dhe_create(struct TLSContext *context) { + _private_tls_ecc_dhe_free(context); + context->ecc_dhe = (ecc_key *)TLS_MALLOC(sizeof(ecc_key)); + memset(context->ecc_dhe, 0, sizeof(ecc_key)); +} + +int tls_set_default_dhe_pg(struct TLSContext *context, const char *p_hex_str, const char *g_hex_str) { + if ((!context) || (context->is_child) || (!context->is_server) || (!p_hex_str) || (!g_hex_str)) + return 0; + + TLS_FREE(context->default_dhe_p); + TLS_FREE(context->default_dhe_g); + + context->default_dhe_p = NULL; + context->default_dhe_g = NULL; + + size_t p_len = strlen(p_hex_str); + size_t g_len = strlen(g_hex_str); + if ((p_len <= 0) || (g_len <= 0)) + return 0; + context->default_dhe_p = (char *)TLS_MALLOC(p_len + 1); + if (!context->default_dhe_p) + return 0; + context->default_dhe_g = (char *)TLS_MALLOC(g_len + 1); + if (!context->default_dhe_g) + return 0; + + memcpy(context->default_dhe_p, p_hex_str, p_len); + context->default_dhe_p[p_len] = 0; + + memcpy(context->default_dhe_g, g_hex_str, g_len); + context->default_dhe_g[g_len] = 0; + return 1; +} +#endif + +const char *tls_alpn(struct TLSContext *context) { + if (!context) + return NULL; + return context->negotiated_alpn; +} + +int tls_add_alpn(struct TLSContext *context, const char *alpn) { + if ((!context) || (!alpn) || (!alpn[0]) || ((context->is_server) && (context->is_child))) + return TLS_GENERIC_ERROR; + int len = strlen(alpn); + if (tls_alpn_contains(context, alpn, len)) + return 0; + context->alpn = (char **)TLS_REALLOC(context->alpn, (context->alpn_count + 1) * sizeof(char *)); + if (!context->alpn) { + context->alpn_count = 0; + return TLS_NO_MEMORY; + } + char *alpn_ref = (char *)TLS_MALLOC(len+1); + context->alpn[context->alpn_count] = alpn_ref; + if (alpn_ref) { + memcpy(alpn_ref, alpn, len); + alpn_ref[len] = 0; + context->alpn_count++; + } else + return TLS_NO_MEMORY; + return 0; +} + +int tls_alpn_contains(struct TLSContext *context, const char *alpn, unsigned char alpn_size) { + if ((!context) || (!alpn) || (!alpn_size)) + return 0; + + if (context->alpn) { + int i; + for (i = 0; i < context->alpn_count; i++) { + const char *alpn_local = context->alpn[i]; + if (alpn_local) { + int len = strlen(alpn_local); + if (alpn_size == len) { + if (!memcmp(alpn_local, alpn, alpn_size)) + return 1; + } + } + } + } + return 0; +} + +void tls_destroy_context(struct TLSContext *context) { + unsigned int i; + if (!context) + return; + if (!context->is_child) { + if (context->certificates) { + for (i = 0; i < context->certificates_count; i++) + tls_destroy_certificate(context->certificates[i]); + } + if (context->root_certificates) { + for (i = 0; i < context->root_count; i++) + tls_destroy_certificate(context->root_certificates[i]); + TLS_FREE(context->root_certificates); + context->root_certificates = NULL; + } + if (context->private_key) + tls_destroy_certificate(context->private_key); +#ifdef TLS_ECDSA_SUPPORTED + if (context->ec_private_key) + tls_destroy_certificate(context->ec_private_key); +#endif + TLS_FREE(context->certificates); +#ifdef TLS_FORWARD_SECRECY + TLS_FREE(context->default_dhe_p); + TLS_FREE(context->default_dhe_g); +#endif + if (context->alpn) { + for (i = 0; i < context->alpn_count; i++) + TLS_FREE(context->alpn[i]); + TLS_FREE(context->alpn); + } + } + if (context->client_certificates) { + for (i = 0; i < context->client_certificates_count; i++) + tls_destroy_certificate(context->client_certificates[i]); + TLS_FREE(context->client_certificates); + } + context->client_certificates = NULL; + TLS_FREE(context->master_key); + TLS_FREE(context->premaster_key); + if (context->crypto.created) + _private_tls_crypto_done(context); + TLS_FREE(context->message_buffer); + _private_tls_done_hash(context, NULL); + _private_tls_destroy_hash(context); + TLS_FREE(context->tls_buffer); + TLS_FREE(context->application_buffer); + // zero out the keys before free + if ((context->exportable_keys) && (context->exportable_size)) + memset(context->exportable_keys, 0, context->exportable_size); + TLS_FREE(context->exportable_keys); + TLS_FREE(context->sni); + TLS_FREE(context->dtls_cookie); + TLS_FREE(context->cached_handshake); +#ifdef TLS_FORWARD_SECRECY + _private_tls_dhe_free(context); + _private_tls_ecc_dhe_free(context); +#endif +#ifdef TLS_ACCEPT_SECURE_RENEGOTIATION + TLS_FREE(context->verify_data); +#endif + TLS_FREE(context->negotiated_alpn); +#ifdef WITH_TLS_13 + TLS_FREE(context->finished_key); + TLS_FREE(context->remote_finished_key); + TLS_FREE(context->server_finished_hash); +#endif +#ifdef TLS_CURVE25519 + TLS_FREE(context->client_secret); +#endif + // DTLS-related buffer + if (context->dtls_data) { + if (context->dtls_data->fragment) { + TLS_FREE(context->dtls_data->fragment->buffer); + TLS_FREE(context->dtls_data->fragment); + } + while (context->dtls_data->dtls_handshake_list) { + struct TLSHandshakeList *next = (struct TLSHandshakeList *)context->dtls_data->dtls_handshake_list->next; + if (context->dtls_data->dtls_handshake_list->msg) { + TLS_FREE(context->dtls_data->dtls_handshake_list->msg); + } + TLS_FREE(context->dtls_data->dtls_handshake_list); + context->dtls_data->dtls_handshake_list = next; + } + if (context->dtls_data->key_exchange) { + TLS_FREE(context->dtls_data->key_exchange); + } + if (context->dtls_data->remote_fingerprint) { + TLS_FREE(context->dtls_data->remote_fingerprint); + } + TLS_FREE(context->dtls_data); + } + TLS_FREE(context); +} + +#ifdef TLS_ACCEPT_SECURE_RENEGOTIATION +void _private_tls_reset_context(struct TLSContext *context) { + unsigned int i; + if (!context) + return; + if (!context->is_child) { + if (context->certificates) { + for (i = 0; i < context->certificates_count; i++) + tls_destroy_certificate(context->certificates[i]); + } + context->certificates = NULL; + if (context->private_key) { + tls_destroy_certificate(context->private_key); + context->private_key = NULL; + } +#ifdef TLS_ECDSA_SUPPORTED + if (context->ec_private_key) { + tls_destroy_certificate(context->ec_private_key); + context->ec_private_key = NULL; + } +#endif + TLS_FREE(context->certificates); + context->certificates = NULL; +#ifdef TLS_FORWARD_SECRECY + TLS_FREE(context->default_dhe_p); + TLS_FREE(context->default_dhe_g); + context->default_dhe_p = NULL; + context->default_dhe_g = NULL; +#endif + } + if (context->client_certificates) { + for (i = 0; i < context->client_certificates_count; i++) + tls_destroy_certificate(context->client_certificates[i]); + TLS_FREE(context->client_certificates); + } + context->client_certificates = NULL; + TLS_FREE(context->master_key); + context->master_key = NULL; + TLS_FREE(context->premaster_key); + context->premaster_key = NULL; + if (context->crypto.created) + _private_tls_crypto_done(context); + _private_tls_done_hash(context, NULL); + _private_tls_destroy_hash(context); + TLS_FREE(context->application_buffer); + context->application_buffer = NULL; + // zero out the keys before free + if ((context->exportable_keys) && (context->exportable_size)) + memset(context->exportable_keys, 0, context->exportable_size); + TLS_FREE(context->exportable_keys); + context->exportable_keys = NULL; + TLS_FREE(context->sni); + context->sni = NULL; + TLS_FREE(context->dtls_cookie); + context->dtls_cookie = NULL; + TLS_FREE(context->cached_handshake); + context->cached_handshake = NULL; + context->connection_status = 0; +#ifdef TLS_FORWARD_SECRECY + _private_tls_dhe_free(context); + _private_tls_ecc_dhe_free(context); +#endif +} +#endif + +int tls_cipher_supported(struct TLSContext *context, unsigned short cipher) { + if (!context) + return 0; + + switch (cipher) { +#ifdef WITH_TLS_13 + case TLS_AES_128_GCM_SHA256: + case TLS_AES_256_GCM_SHA384: + case TLS_CHACHA20_POLY1305_SHA256: + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + return 1; + return 0; +#endif +#ifdef TLS_FORWARD_SECRECY +#ifdef TLS_ECDSA_SUPPORTED + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA: +#ifdef TLS_CLIENT_ECDSA + if ((context) && (((context->certificates) && (context->certificates_count) && (context->ec_private_key)) || (!context->is_server))) +#else + if ((context) && (context->certificates) && (context->certificates_count) && (context->ec_private_key)) +#endif + return 1; + return 0; + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: +#ifdef TLS_WITH_CHACHA20_POLY1305 + case TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256: +#endif + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) { +#ifdef TLS_CLIENT_ECDSA + if ((context) && (((context->certificates) && (context->certificates_count) && (context->ec_private_key)) || (!context->is_server))) +#else + if ((context) && (context->certificates) && (context->certificates_count) && (context->ec_private_key)) +#endif + return 1; + } + return 0; +#endif + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA: +#endif + case TLS_RSA_WITH_AES_128_CBC_SHA: + case TLS_RSA_WITH_AES_256_CBC_SHA: + return 1; +#ifdef TLS_FORWARD_SECRECY + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_DHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: +#ifdef TLS_WITH_CHACHA20_POLY1305 + case TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256: +#endif +#endif + case TLS_RSA_WITH_AES_128_GCM_SHA256: + case TLS_RSA_WITH_AES_128_CBC_SHA256: + case TLS_RSA_WITH_AES_256_CBC_SHA256: + case TLS_RSA_WITH_AES_256_GCM_SHA384: + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) + return 1; + return 0; + } + return 0; +} + +int tls_cipher_is_fs(struct TLSContext *context, unsigned short cipher) { + if (!context) + return 0; +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + switch (cipher) { + case TLS_AES_128_GCM_SHA256: + case TLS_AES_256_GCM_SHA384: + case TLS_CHACHA20_POLY1305_SHA256: + return 1; + } + return 0; + } +#endif + switch (cipher) { +#ifdef TLS_ECDSA_SUPPORTED + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA: +#ifdef TLS_WITH_CHACHA20_POLY1305 + case TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256: +#endif + if ((context) && (context->certificates) && (context->certificates_count) && (context->ec_private_key)) + return 1; + return 0; + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) { + if ((context) && (context->certificates) && (context->certificates_count) && (context->ec_private_key)) + return 1; + } + return 0; +#endif + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA: + return 1; + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_DHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: +#ifdef TLS_WITH_CHACHA20_POLY1305 + case TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256: +#endif + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) + return 1; + break; + } + return 0; +} + +#ifdef WITH_KTLS +int _private_tls_prefer_ktls(struct TLSContext *context, unsigned short cipher) { + if ((context->version == TLS_V13) || (context->version == DTLS_V13) || ((context->version != TLS_V12) && (context->version != DTLS_V12))) + return 0; + + switch (cipher) { + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + if ((context->version == TLS_V13) || (context->version == DTLS_V13) || (context->version == TLS_V12) || (context->version == DTLS_V12)) { + if ((context->certificates) && (context->certificates_count) && (context->ec_private_key)) + return 1; + } + break; + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + return 1; + } + return 0; +} +#endif + +int tls_choose_cipher(struct TLSContext *context, const unsigned char *buf, int buf_len, int *scsv_set) { + int i; + if (scsv_set) + *scsv_set = 0; + if (!context) + return 0; + int selected_cipher = TLS_NO_COMMON_CIPHER; +#ifdef TLS_FORWARD_SECRECY +#ifdef WITH_KTLS + for (i = 0; i < buf_len; i+=2) { + unsigned short cipher = ntohs(*(unsigned short *)&buf[i]); + if (_private_tls_prefer_ktls(context, cipher)) { + selected_cipher = cipher; + break; + } + } +#endif + if (selected_cipher == TLS_NO_COMMON_CIPHER) { + for (i = 0; i < buf_len; i+=2) { + unsigned short cipher = ntohs(*(unsigned short *)&buf[i]); + if (tls_cipher_is_fs(context, cipher)) { + selected_cipher = cipher; + break; + } + } + } +#endif + for (i = 0; i < buf_len; i+=2) { + unsigned short cipher = ntohs(*(unsigned short *)&buf[i]); + if (cipher == TLS_FALLBACK_SCSV) { + if (scsv_set) + *scsv_set = 1; + if (selected_cipher != TLS_NO_COMMON_CIPHER) + break; + } +#ifndef TLS_ROBOT_MITIGATION + else + if ((selected_cipher == TLS_NO_COMMON_CIPHER) && (tls_cipher_supported(context, cipher))) + selected_cipher = cipher; +#endif + } + return selected_cipher; +} + +int tls_cipher_is_ephemeral(struct TLSContext *context) { + if (context) { + switch (context->cipher) { + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA: + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA256: + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_DHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + return 1; + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256: + return 2; +#ifdef WITH_TLS_13 + case TLS_AES_128_GCM_SHA256: + case TLS_CHACHA20_POLY1305_SHA256: + case TLS_AES_128_CCM_SHA256: + case TLS_AES_128_CCM_8_SHA256: + case TLS_AES_256_GCM_SHA384: + if (context->dhe) + return 1; + return 2; +#endif + } + } + return 0; +} + +const char *tls_cipher_name(struct TLSContext *context) { + if (context) { + switch (context->cipher) { + case TLS_RSA_WITH_AES_128_CBC_SHA: + return "RSA-AES128CBC-SHA"; + case TLS_RSA_WITH_AES_256_CBC_SHA: + return "RSA-AES256CBC-SHA"; + case TLS_RSA_WITH_AES_128_CBC_SHA256: + return "RSA-AES128CBC-SHA256"; + case TLS_RSA_WITH_AES_256_CBC_SHA256: + return "RSA-AES256CBC-SHA256"; + case TLS_RSA_WITH_AES_128_GCM_SHA256: + return "RSA-AES128GCM-SHA256"; + case TLS_RSA_WITH_AES_256_GCM_SHA384: + return "RSA-AES256GCM-SHA384"; + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA: + return "DHE-RSA-AES128CBC-SHA"; + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA: + return "DHE-RSA-AES256CBC-SHA"; + case TLS_DHE_RSA_WITH_AES_128_CBC_SHA256: + return "DHE-RSA-AES128CBC-SHA256"; + case TLS_DHE_RSA_WITH_AES_256_CBC_SHA256: + return "DHE-RSA-AES256CBC-SHA256"; + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + return "DHE-RSA-AES128GCM-SHA256"; + case TLS_DHE_RSA_WITH_AES_256_GCM_SHA384: + return "DHE-RSA-AES256GCM-SHA384"; + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA: + return "ECDHE-RSA-AES128CBC-SHA"; + case TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA: + return "ECDHE-RSA-AES256CBC-SHA"; + case TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256: + return "ECDHE-RSA-AES128CBC-SHA256"; + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + return "ECDHE-RSA-AES128GCM-SHA256"; + case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: + return "ECDHE-RSA-AES256GCM-SHA384"; + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA: + return "ECDHE-ECDSA-AES128CBC-SHA"; + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA: + return "ECDHE-ECDSA-AES256CBC-SHA"; + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256: + return "ECDHE-ECDSA-AES128CBC-SHA256"; + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384: + return "ECDHE-ECDSA-AES256CBC-SHA384"; + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + return "ECDHE-ECDSA-AES128GCM-SHA256"; + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: + return "ECDHE-ECDSA-AES256GCM-SHA384"; + case TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + return "ECDHE-RSA-CHACHA20-POLY1305-SHA256"; + case TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256: + return "ECDHE-ECDSA-CHACHA20-POLY1305-SHA256"; + case TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256: + return "ECDHE-DHE-CHACHA20-POLY1305-SHA256"; + case TLS_AES_128_GCM_SHA256: + return "TLS-AES-128-GCM-SHA256"; + case TLS_AES_256_GCM_SHA384: + return "TLS-AES-256-GCM-SHA384"; + case TLS_CHACHA20_POLY1305_SHA256: + return "TLS-CHACHA20-POLY1305-SHA256"; + case TLS_AES_128_CCM_SHA256: + return "TLS-AES-128-CCM-SHA256"; + case TLS_AES_128_CCM_8_SHA256: + return "TLS-AES-128-CCM-8-SHA256"; + } + } + return "UNKNOWN"; +} + +#ifdef TLS_FORWARD_SECRECY +int _private_tls_dh_export_Y(unsigned char *Ybuf, unsigned long *Ylen, DHKey *key) { + unsigned long len; + + if ((Ybuf == NULL) || (Ylen == NULL) || (key == NULL)) + return TLS_GENERIC_ERROR; + + len = mp_unsigned_bin_size(key->y); + if (len > *Ylen) + return TLS_GENERIC_ERROR; + + *Ylen = len; + return 0; + } + +int _private_tls_dh_export_pqY(unsigned char *pbuf, unsigned long *plen, unsigned char *gbuf, unsigned long *glen, unsigned char *Ybuf, unsigned long *Ylen, DHKey *key) { + unsigned long len; + int err; + + if ((pbuf == NULL) || (plen == NULL) || (gbuf == NULL) || (glen == NULL) || (Ybuf == NULL) || (Ylen == NULL) || (key == NULL)) + return TLS_GENERIC_ERROR; + + len = mp_unsigned_bin_size(key->y); + if (len > *Ylen) + return TLS_GENERIC_ERROR; + + if ((err = mp_to_unsigned_bin(key->y, Ybuf)) != CRYPT_OK) + return err; + + *Ylen = len; + + len = mp_unsigned_bin_size(key->p); + if (len > *plen) + return TLS_GENERIC_ERROR; + + if ((err = mp_to_unsigned_bin(key->p, pbuf)) != CRYPT_OK) + return err; + + *plen = len; + + len = mp_unsigned_bin_size(key->g); + if (len > *glen) + return TLS_GENERIC_ERROR; + + if ((err = mp_to_unsigned_bin(key->g, gbuf)) != CRYPT_OK) + return err; + + *glen = len; + + return 0; +} + +void _private_tls_dh_clear_key(DHKey *key) { + mp_clear_multi(key->g, key->p, key->x, key->y, NULL); + key->g = NULL; + key->p = NULL; + key->x = NULL; + key->y = NULL; +} + +int _private_tls_dh_make_key(int keysize, DHKey *key, const char *pbuf, const char *gbuf, int pbuf_len, int gbuf_len) { + unsigned char *buf; + int err; + if (!key) + return TLS_GENERIC_ERROR; + + static prng_state prng; + int wprng = find_prng("sprng"); + if ((err = prng_is_valid(wprng)) != CRYPT_OK) + return err; + + buf = (unsigned char *)TLS_MALLOC(keysize); + if (!buf) + return TLS_NO_MEMORY; + + if (rng_make_prng(keysize, wprng, &prng, NULL) != CRYPT_OK) { + TLS_FREE(buf); + return TLS_GENERIC_ERROR; + } + + if (prng_descriptor[wprng].read(buf, keysize, &prng) != (unsigned long)keysize) { + TLS_FREE(buf); + return TLS_GENERIC_ERROR; + } + + if ((err = mp_init_multi(&key->g, &key->p, &key->x, &key->y, NULL)) != CRYPT_OK) { + TLS_FREE(buf); + + return TLS_GENERIC_ERROR; + } + + if (gbuf_len <= 0) { + if ((err = mp_read_radix(key->g, gbuf, 16)) != CRYPT_OK) { + TLS_FREE(buf); + _private_tls_dh_clear_key(key); + return TLS_GENERIC_ERROR; + } + } else { + if ((err = mp_read_unsigned_bin(key->g, (unsigned char *)gbuf, gbuf_len)) != CRYPT_OK) { + TLS_FREE(buf); + _private_tls_dh_clear_key(key); + return TLS_GENERIC_ERROR; + } + } + + if (pbuf_len <= 0) { + if ((err = mp_read_radix(key->p, pbuf, 16)) != CRYPT_OK) { + TLS_FREE(buf); + _private_tls_dh_clear_key(key); + return TLS_GENERIC_ERROR; + } + } else { + if ((err = mp_read_unsigned_bin(key->p, (unsigned char *)pbuf, pbuf_len)) != CRYPT_OK) { + TLS_FREE(buf); + _private_tls_dh_clear_key(key); + return TLS_GENERIC_ERROR; + } + } + + if ((err = mp_read_unsigned_bin(key->x, buf, keysize)) != CRYPT_OK) { + TLS_FREE(buf); + _private_tls_dh_clear_key(key); + return TLS_GENERIC_ERROR; + } + + if ((err = mp_exptmod(key->g, key->x, key->p, key->y)) != CRYPT_OK) { + TLS_FREE(buf); + _private_tls_dh_clear_key(key); + return TLS_GENERIC_ERROR; + } + + TLS_FREE(buf); + return 0; +} +#endif + +int tls_is_ecdsa(struct TLSContext *context) { + if (!context) + return 0; + switch (context->cipher) { + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA: + case TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: +#ifdef TLS_WITH_CHACHA20_POLY1305 + case TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256: +#endif + return 1; + } +#ifdef WITH_TLS_13 + if (context->ec_private_key) + return 1; +#endif + return 0; +} + +struct TLSPacket *tls_build_client_key_exchange(struct TLSContext *context) { + if (context->is_server) { + DEBUG_PRINT("CANNOT BUILD CLIENT KEY EXCHANGE MESSAGE FOR SERVERS\n"); + return NULL; + } + + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, context->version, 0); + tls_packet_uint8(packet, 0x10); +#ifdef TLS_FORWARD_SECRECY + int ephemeral = tls_cipher_is_ephemeral(context); + if ((ephemeral) && (context->premaster_key) && (context->premaster_key_len)) { + if (ephemeral == 1) { + unsigned char dh_Ys[0xFFF]; + unsigned char dh_p[0xFFF]; + unsigned char dh_g[0xFFF]; + unsigned long dh_p_len = sizeof(dh_p); + unsigned long dh_g_len = sizeof(dh_g); + unsigned long dh_Ys_len = sizeof(dh_Ys); + + if (_private_tls_dh_export_pqY(dh_p, &dh_p_len, dh_g, &dh_g_len, dh_Ys, &dh_Ys_len, context->dhe)) { + DEBUG_PRINT("ERROR EXPORTING DHE KEY %p\n", context->dhe); + TLS_FREE(packet); + _private_tls_dhe_free(context); + return NULL; + } + _private_tls_dhe_free(context); + DEBUG_DUMP_HEX_LABEL("Yc", dh_Ys, dh_Ys_len); + tls_packet_uint24(packet, dh_Ys_len + 2); + if (context->dtls) + _private_dtls_handshake_data(context, packet, dh_Ys_len + 2); + tls_packet_uint16(packet, dh_Ys_len); + tls_packet_append(packet, dh_Ys, dh_Ys_len); + } else + if (context->ecc_dhe) { + unsigned char out[TLS_MAX_RSA_KEY]; + unsigned long out_len = TLS_MAX_RSA_KEY; + + if (ecc_ansi_x963_export(context->ecc_dhe, out, &out_len)) { + DEBUG_PRINT("Error exporting ECC key\n"); + TLS_FREE(packet); + return NULL; + } + _private_tls_ecc_dhe_free(context); + tls_packet_uint24(packet, out_len + 1); + if (context->dtls) { + _private_dtls_handshake_data(context, packet, out_len + 1); + context->dtls_seq ++; + } + tls_packet_uint8(packet, out_len); + tls_packet_append(packet, out, out_len); + } +#ifdef TLS_CURVE25519 + else + if ((context->curve == &x25519) && (context->client_secret)) { + static const unsigned char basepoint[32] = {9}; + unsigned char shared_key[32]; + curve25519(shared_key, context->client_secret, basepoint); + tls_packet_uint24(packet, 32 + 1); + tls_packet_uint8(packet, 32); + tls_packet_append(packet, shared_key, 32); + TLS_FREE(context->client_secret); + context->client_secret = NULL; + } +#endif + _private_tls_compute_key(context, 48); + } else +#endif + _private_tls_build_random(packet); + context->connection_status = 2; + tls_packet_update(packet); + return packet; +} + +void _private_dtls_handshake_data(struct TLSContext *context, struct TLSPacket *packet, unsigned int framelength) { + // message seq + tls_packet_uint16(packet, context->dtls_seq); + // fragment offset + tls_packet_uint24(packet, 0); + // fragment length + tls_packet_uint24(packet, framelength); +} + +void _private_dtls_handshake_copyframesize(struct TLSPacket *packet) { + packet->buf[22] = packet->buf[14]; + packet->buf[23] = packet->buf[15]; + packet->buf[24] = packet->buf[16]; +} + +struct TLSPacket *tls_build_server_key_exchange(struct TLSContext *context, int method) { + if (!context->is_server) { + DEBUG_PRINT("CANNOT BUILD SERVER KEY EXCHANGE MESSAGE FOR CLIENTS\n"); + return NULL; + } + + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, context->version, 0); + if ((context->dtls) && (context->dtls_data->key_exchange) && (context->dtls_data->key_exchange_len > packet->len)) { + tls_packet_append(packet, context->dtls_data->key_exchange, context->dtls_data->key_exchange_len); + tls_packet_update(packet); + context->dtls_seq ++; + return packet; + } + int packet_offset = packet->len; + tls_packet_uint8(packet, 0x0C); + unsigned char dummy[3]; + tls_packet_append(packet, dummy, 3); + if (context->dtls) + _private_dtls_handshake_data(context, packet, 0); + int start_len = packet->len; +#ifdef TLS_FORWARD_SECRECY + if (method == KEA_dhe_rsa) { + + if (!context->dhe) { + tls_init(); + _private_tls_dhe_create(context); + + const char *default_dhe_p = context->default_dhe_p; + const char *default_dhe_g = context->default_dhe_g; + int key_size; + if ((!default_dhe_p) || (!default_dhe_g)) { + default_dhe_p = TLS_DH_DEFAULT_P; + default_dhe_g = TLS_DH_DEFAULT_G; + key_size = TLS_DHE_KEY_SIZE / 8; + } else { + key_size = strlen(default_dhe_p); + } + + if (_private_tls_dh_make_key(key_size, context->dhe, default_dhe_p, default_dhe_g, 0, 0)) { + DEBUG_PRINT("ERROR CREATING DHE KEY\n"); + TLS_FREE(packet); + TLS_FREE(context->dhe); + context->dhe = NULL; + return NULL; + } + } + + unsigned char dh_Ys[0xFFF]; + unsigned char dh_p[0xFFF]; + unsigned char dh_g[0xFFF]; + unsigned long dh_p_len = sizeof(dh_p); + unsigned long dh_g_len = sizeof(dh_g); + unsigned long dh_Ys_len = sizeof(dh_Ys); + + if (_private_tls_dh_export_pqY(dh_p, &dh_p_len, dh_g, &dh_g_len, dh_Ys, &dh_Ys_len, context->dhe)) { + DEBUG_PRINT("ERROR EXPORTING DHE KEY\n"); + TLS_FREE(packet); + return NULL; + } + + DEBUG_PRINT("LEN: %lu (%lu, %lu)\n", dh_Ys_len, dh_p_len, dh_g_len); + DEBUG_DUMP_HEX_LABEL("DHE PK", dh_Ys, dh_Ys_len); + DEBUG_DUMP_HEX_LABEL("DHE P", dh_p, dh_p_len); + DEBUG_DUMP_HEX_LABEL("DHE G", dh_g, dh_g_len); + + tls_packet_uint16(packet, dh_p_len); + tls_packet_append(packet, dh_p, dh_p_len); + + tls_packet_uint16(packet, dh_g_len); + tls_packet_append(packet, dh_g, dh_g_len); + + tls_packet_uint16(packet, dh_Ys_len); + tls_packet_append(packet, dh_Ys, dh_Ys_len); + //dh_p + //dh_g + //dh_Ys + } else + if (method == KEA_ec_diffie_hellman) { + // 3 = named curve + if (!context->curve) + context->curve = default_curve; + tls_packet_uint8(packet, 3); + tls_packet_uint16(packet, context->curve->iana); + if (!context->ecc_dhe) { + tls_init(); + _private_tls_ecc_dhe_create(context); + + ltc_ecc_set_type *dp = (ltc_ecc_set_type *)&context->curve->dp; + + if (ecc_make_key_ex(NULL, find_prng("sprng"), context->ecc_dhe, dp)) { + TLS_FREE(context->ecc_dhe); + context->ecc_dhe = NULL; + DEBUG_PRINT("Error generating ECC key\n"); + TLS_FREE(packet); + return NULL; + } + } + unsigned char out[TLS_MAX_RSA_KEY]; + unsigned long out_len = TLS_MAX_RSA_KEY; + if (ecc_ansi_x963_export(context->ecc_dhe, out, &out_len)) { + DEBUG_PRINT("Error exporting ECC key\n"); + TLS_FREE(packet); + return NULL; + } + tls_packet_uint8(packet, out_len); + tls_packet_append(packet, out, out_len); + } else +#endif + { + TLS_FREE(packet); + DEBUG_PRINT("Unsupported ephemeral method: %i\n", method); + return NULL; + } + + // signature + unsigned int params_len = packet->len - start_len; + unsigned int message_len = params_len + TLS_CLIENT_RANDOM_SIZE + TLS_SERVER_RANDOM_SIZE; + unsigned char *message = (unsigned char *)TLS_MALLOC(message_len); + if (message) { + unsigned char out[TLS_MAX_RSA_KEY]; + unsigned long out_len = TLS_MAX_RSA_KEY; + + int hash_algorithm; + if ((context->version != TLS_V13) && (context->version != DTLS_V13) && (context->version != TLS_V12) && (context->version != DTLS_V12)) { + hash_algorithm = _md5_sha1; + } else { + if ((context->version == TLS_V13) || (context->version == DTLS_V13) || (context->version == TLS_V12) || (context->version == DTLS_V12)) + hash_algorithm = sha256; + else + hash_algorithm = sha1; + +#ifdef TLS_ECDSA_SUPPORTED + if (tls_is_ecdsa(context)) { + if ((context->version == TLS_V13) || (context->version == DTLS_V13) || (context->version == TLS_V12)) + hash_algorithm = sha512; + tls_packet_uint8(packet, hash_algorithm); + tls_packet_uint8(packet, ecdsa); + } else +#endif + { + tls_packet_uint8(packet, hash_algorithm); + tls_packet_uint8(packet, rsa_sign); + } + } + + memcpy(message, context->remote_random, TLS_CLIENT_RANDOM_SIZE); + memcpy(message + TLS_CLIENT_RANDOM_SIZE, context->local_random, TLS_SERVER_RANDOM_SIZE); + memcpy(message + TLS_CLIENT_RANDOM_SIZE + TLS_SERVER_RANDOM_SIZE, packet->buf + start_len, params_len); +#ifdef TLS_ECDSA_SUPPORTED + if (tls_is_ecdsa(context)) { + if (_private_tls_sign_ecdsa(context, hash_algorithm, message, message_len, out, &out_len) == 1) { + DEBUG_PRINT("Signing OK! (ECDSA, length %lu)\n", out_len); + tls_packet_uint16(packet, out_len); + tls_packet_append(packet, out, out_len); + } + } else +#endif + if (_private_tls_sign_rsa(context, hash_algorithm, message, message_len, out, &out_len) == 1) { + DEBUG_PRINT("Signing OK! (length %lu)\n", out_len); + tls_packet_uint16(packet, out_len); + tls_packet_append(packet, out, out_len); + } + TLS_FREE(message); + } + if ((!packet->broken) && (packet->buf)) { + int remaining = packet->len - start_len; + int payload_pos = 6; + if (context->dtls) + payload_pos = 14; + packet->buf[payload_pos] = remaining / 0x10000; + remaining %= 0x10000; + packet->buf[payload_pos + 1] = remaining / 0x100; + remaining %= 0x100; + packet->buf[payload_pos + 2] = remaining; + if (context->dtls) { + _private_dtls_handshake_copyframesize(packet); + context->dtls_seq ++; + } + } + tls_packet_update(packet); + if (context->dtls_data) { + if (context->dtls_data->key_exchange) { + TLS_FREE(context->dtls_data->key_exchange); + } + context->dtls_data->key_exchange = (unsigned char *)TLS_MALLOC(packet->len - packet_offset); + if (context->dtls_data->key_exchange) { + context->dtls_data->key_exchange_len = packet->len - packet_offset; + memcpy(context->dtls_data->key_exchange, packet->buf + packet_offset, context->dtls_data->key_exchange_len); + } + } + return packet; +} + +void _private_tls_set_session_id(struct TLSContext *context) { + if (((context->version == TLS_V13) || (context->version == DTLS_V13)) && (context->session_size == TLS_MAX_SESSION_ID)) + return; + if (tls_random(context->session, TLS_MAX_SESSION_ID)) + context->session_size = TLS_MAX_SESSION_ID; + else + context->session_size = 0; +} + +struct TLSPacket *tls_build_hello(struct TLSContext *context, int tls13_downgrade) { + tls_init(); +#ifdef WITH_TLS_13 + if (context->connection_status == 4) { + static unsigned char sha256_helloretryrequest[] = {0xCF, 0x21, 0xAD, 0x74, 0xE5, 0x9A, 0x61, 0x11, 0xBE, 0x1D, 0x8C, 0x02, 0x1E, 0x65, 0xB8, 0x91, 0xC2, 0xA2, 0x11, 0x16, 0x7A, 0xBB, 0x8C, 0x5E, 0x07, 0x9E, 0x09, 0xE2, 0xC8, 0xA8, 0x33, 0x9C}; + memcpy(context->local_random, sha256_helloretryrequest, 32); + unsigned char header[4] = {0xFE, 0, 0, 0}; + unsigned char hash[TLS_MAX_SHA_SIZE ]; + int hash_len = _private_tls_done_hash(context, hash); + header[3] = (unsigned char)hash_len; + _private_tls_update_hash(context, header, sizeof(header), 1, 0); + _private_tls_update_hash(context, hash, hash_len, 1, 0); + } else + if ((!context->is_server) || ((context->version != TLS_V13) && (context->version != DTLS_V13))) +#endif + if ((!context->dtls) || (!context->dtls_data->has_random)) { + if (!tls_random(context->local_random, context->is_server ? TLS_SERVER_RANDOM_SIZE : TLS_CLIENT_RANDOM_SIZE)) + return NULL; + // if (!context->is_server) + *(unsigned int *)context->local_random = htonl((unsigned int)time(NULL)); + + if (context->dtls) + context->dtls_data->has_random = 1; + } + + if ((context->is_server) && (tls13_downgrade)) { + if ((tls13_downgrade == TLS_V12) || (tls13_downgrade == DTLS_V12)) + memcpy(context->local_random + TLS_SERVER_RANDOM_SIZE - 8, "DOWNGRD\x01", 8); + else + memcpy(context->local_random + TLS_SERVER_RANDOM_SIZE - 8, "DOWNGRD\x00", 8); + } + unsigned short packet_version = context->version; + unsigned short version = context->version; +#ifdef WITH_TLS_13 + if (context->version == TLS_V13) + version = TLS_V12; + else + if (context->version == DTLS_V13) + version = DTLS_V12; +#endif + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, version, 0); + if (packet) { + // hello + if (context->is_server) + tls_packet_uint8(packet, 0x02); + else + tls_packet_uint8(packet, 0x01); + unsigned char dummy[3]; + tls_packet_append(packet, dummy, 3); + + if (context->dtls) + _private_dtls_handshake_data(context, packet, 0); + + int start_len = packet->len; + tls_packet_uint16(packet, version); + if (context->is_server) + tls_packet_append(packet, context->local_random, TLS_SERVER_RANDOM_SIZE); + else + tls_packet_append(packet, context->local_random, TLS_CLIENT_RANDOM_SIZE); + +#ifdef IGNORE_SESSION_ID + // session size + tls_packet_uint8(packet, 0); +#else + if ((!context->dtls) || (!context->session_size)) + _private_tls_set_session_id(context); + // session size + tls_packet_uint8(packet, context->session_size); + if (context->session_size) + tls_packet_append(packet, context->session, context->session_size); +#endif + + int extension_len = 0; + int alpn_len = 0; + int alpn_negotiated_len = 0; + int i; +#ifdef WITH_TLS_13 + unsigned char shared_key[TLS_MAX_RSA_KEY]; + unsigned long shared_key_len = TLS_MAX_RSA_KEY; + unsigned short shared_key_short = 0; + int selected_group = 0; + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + if (context->connection_status == 4) { + // connection_status == 4 => hello retry request + extension_len += 6; + } else + if (context->is_server) { +#ifdef TLS_CURVE25519 + if (context->curve == &x25519) { + extension_len += 8 + 32; + shared_key_short = (unsigned short)32; + if (context->finished_key) { + memcpy(shared_key, context->finished_key, 32); + TLS_FREE(context->finished_key); + context->finished_key = NULL; + } + selected_group = context->curve->iana; + // make context->curve NULL (x25519 is a different implementation) + context->curve = NULL; + } else +#endif + if (context->ecc_dhe) { + if (ecc_ansi_x963_export(context->ecc_dhe, shared_key, &shared_key_len)) { + DEBUG_PRINT("Error exporting ECC DHE key\n"); + tls_destroy_packet(packet); + return tls_build_alert(context, 1, internal_error); + } + _private_tls_ecc_dhe_free(context); + extension_len += 8 + shared_key_len; + shared_key_short = (unsigned short)shared_key_len; + if (context->curve) + selected_group = context->curve->iana; + } else + if (context->dhe) { + selected_group = context->dhe->iana; + _private_tls_dh_export_Y(shared_key, &shared_key_len, context->dhe); + _private_tls_dhe_free(context); + extension_len += 8 + shared_key_len; + shared_key_short = (unsigned short)shared_key_len; + } + } + // supported versions + if (context->is_server) + extension_len += 6; + else + extension_len += 9; + } + if ((context->is_server) && (context->negotiated_alpn) && (context->version != TLS_V13) && (context->version != DTLS_V13)) { +#else + if ((context->is_server) && (context->negotiated_alpn)) { +#endif + alpn_negotiated_len = strlen(context->negotiated_alpn); + alpn_len = alpn_negotiated_len + 1; + extension_len += alpn_len + 6; + } else + if ((!context->is_server) && (context->alpn_count)) { + for (i = 0; i < context->alpn_count;i++) { + if (context->alpn[i]) { + int len = strlen(context->alpn[i]); + if (len) + alpn_len += len + 1; + } + } + if (alpn_len) + extension_len += alpn_len + 6; + } + + // ciphers + if (context->is_server) { + // fallback ... this should never happen + if (!context->cipher) + context->cipher = TLS_DHE_RSA_WITH_AES_128_CBC_SHA; + + tls_packet_uint16(packet, context->cipher); + // no compression + tls_packet_uint8(packet, 0); +#ifndef STRICT_TLS + if ((context->version == TLS_V13) || (context->version == DTLS_V13) || (context->version == TLS_V12) || (context->version == DTLS_V12)) { + // extensions size +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + tls_packet_uint16(packet, extension_len); + } else +#endif + { + if (context->dtls == 4) { + // use_srtp + extension_len += 9; + // record_size_limit + // extension_len += 6; +#ifdef TLS_DTLS_EXTENDED_MASTER_SECRET + if ((context->dtls) && (context->dtls_data) && (context->dtls_data->extended_master_secret)) + extension_len += 4; +#endif + } + + tls_packet_uint16(packet, 5 + extension_len); + // secure renegotation + // advertise it, but refuse renegotiation + tls_packet_uint16(packet, 0xff01); +#ifdef TLS_ACCEPT_SECURE_RENEGOTIATION + // a little defensive + if ((context->verify_len) && (!context->verify_data)) + context->verify_len = 0; + tls_packet_uint16(packet, context->verify_len + 1); + tls_packet_uint8(packet, context->verify_len); + if (context->verify_len) + tls_packet_append(packet, (unsigned char *)context->verify_data, context->verify_len); +#else + tls_packet_uint16(packet, 1); + tls_packet_uint8(packet, 0); +#endif + } + if (alpn_len) { + tls_packet_uint16(packet, 0x10); + tls_packet_uint16(packet, alpn_len + 2); + tls_packet_uint16(packet, alpn_len); + + tls_packet_uint8(packet, alpn_negotiated_len); + tls_packet_append(packet, (unsigned char *)context->negotiated_alpn, alpn_negotiated_len); + } + if (context->dtls == 4) { +#ifdef TLS_DTLS_EXTENDED_MASTER_SECRET + if ((context->dtls) && (context->dtls_data) && (context->dtls_data->extended_master_secret)) { + tls_packet_uint16(packet, 0x17); + tls_packet_uint16(packet, 0); + } +#endif + // record_size_limit + // tls_packet_uint16(packet, 0x1C); + // tls_packet_uint16(packet, 2); + // tls_packet_uint16(packet, 0x4000); + + tls_packet_uint16(packet, 0x0E); + tls_packet_uint16(packet, 5); + tls_packet_uint16(packet, 2); + tls_packet_uint16(packet, SRTP_AES128_CM_HMAC_SHA1_80); + tls_packet_uint8(packet, 0); + } + } +#endif + } else { + if (context->dtls) { + tls_packet_uint8(packet, context->dtls_cookie_len); + if (context->dtls_cookie_len) + tls_packet_append(packet, context->dtls_cookie, context->dtls_cookie_len); + } + +#ifndef STRICT_TLS +#ifdef WITH_TLS_13 +#ifdef TLS_FORWARD_SECRECY + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + #ifdef TLS_WITH_CHACHA20_POLY1305 + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(9, 0)); + tls_packet_uint16(packet, TLS_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_AES_256_GCM_SHA384); + tls_packet_uint16(packet, TLS_CHACHA20_POLY1305_SHA256); + #else + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(8, 0)); + tls_packet_uint16(packet, TLS_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_AES_256_GCM_SHA384); + #endif + #ifdef TLS_PREFER_CHACHA20 + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_128_GCM_SHA256); + #else + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256); + #endif + } else +#endif +#endif + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) { +#endif +#ifdef TLS_FORWARD_SECRECY +#ifdef TLS_CLIENT_ECDHE +#ifdef TLS_WITH_CHACHA20_POLY1305 + #ifdef TLS_CLIENT_ECDSA + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(16, 5)); + #ifdef TLS_PREFER_CHACHA20 + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256); + #endif + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256); + #ifndef TLS_PREFER_CHACHA20 + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256); + #endif + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA); + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA); + #else + // sizeof ciphers (16 ciphers * 2 bytes) + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(11, 5)); + #endif +#else + #ifdef TLS_CLIENT_ECDSA + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(13, 5)); + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA); + tls_packet_uint16(packet, TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA); + #else + // sizeof ciphers (14 ciphers * 2 bytes) + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(9, 5)); + #endif +#endif +#ifdef TLS_WITH_CHACHA20_POLY1305 + #ifdef TLS_PREFER_CHACHA20 + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256); + #endif +#endif + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256); +#ifdef TLS_WITH_CHACHA20_POLY1305 + #ifndef TLS_PREFER_CHACHA20 + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256); + #endif +#endif +#else +#ifdef TLS_WITH_CHACHA20_POLY1305 + // sizeof ciphers (11 ciphers * 2 bytes) + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(6, 5)); +#else + // sizeof ciphers (10 ciphers * 2 bytes) + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(5, 5)); +#endif +#endif + // not yet supported, because the first message sent (this one) + // is already hashed by the client with sha256 (sha384 not yet supported client-side) + // but is fully suported server-side + // tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_256_GCM_SHA384); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_256_CBC_SHA256); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_128_CBC_SHA256); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_256_CBC_SHA); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_128_CBC_SHA); +#ifdef TLS_WITH_CHACHA20_POLY1305 + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256); +#endif +#else + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(0, 5)); +#endif + // tls_packet_uint16(packet, TLS_RSA_WITH_AES_256_GCM_SHA384); +#ifndef TLS_ROBOT_MITIGATION + tls_packet_uint16(packet, TLS_RSA_WITH_AES_128_GCM_SHA256); + tls_packet_uint16(packet, TLS_RSA_WITH_AES_256_CBC_SHA256); + tls_packet_uint16(packet, TLS_RSA_WITH_AES_128_CBC_SHA256); + tls_packet_uint16(packet, TLS_RSA_WITH_AES_256_CBC_SHA); + tls_packet_uint16(packet, TLS_RSA_WITH_AES_128_CBC_SHA); +#endif +#ifndef STRICT_TLS + } else { +#ifdef TLS_FORWARD_SECRECY +#ifdef TLS_CLIENT_ECDHE + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(5, 2)); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA); + tls_packet_uint16(packet, TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA); +#else + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(3, 2)); +#endif + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_256_CBC_SHA); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_256_CBC_SHA); + tls_packet_uint16(packet, TLS_DHE_RSA_WITH_AES_128_CBC_SHA); +#else + tls_packet_uint16(packet, TLS_CIPHERS_SIZE(0, 2)); +#endif +#ifndef TLS_ROBOT_MITIGATION + tls_packet_uint16(packet, TLS_RSA_WITH_AES_256_CBC_SHA); + tls_packet_uint16(packet, TLS_RSA_WITH_AES_128_CBC_SHA); +#endif + } +#endif + // compression + tls_packet_uint8(packet, 1); + // no compression + tls_packet_uint8(packet, 0); + + if ((context->version == TLS_V12) || (context->version == DTLS_V12) || (context->version == TLS_V13) || (context->version == DTLS_V13)) { + if (context->dtls == 4) { + // use_srtp + extension_len += 9;// 15; + } + + int sni_len = 0; + if (context->sni) + sni_len = strlen(context->sni); + +#ifdef TLS_CLIENT_ECDHE + extension_len += 12; +#endif + if (sni_len) + extension_len += sni_len + 9; +#ifdef WITH_TLS_13 + if ((!context->is_server) && ((context->version == TLS_V13) || (context->version == DTLS_V13))) { +#ifdef TLS_CURVE25519 + extension_len += 70; +#else + // secp256r1 produces 65 bytes export + extension_len += 103; +#endif + } +#endif + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) { + // signature algorithms + extension_len += 6 + 2 * TLS_HASH_ALGO_NUMBER * TLS_SIGN_ALGO_NUMBER; + } + + tls_packet_uint16(packet, extension_len); + + if (sni_len) { + // sni extension + tls_packet_uint16(packet, 0x00); + // sni extension len + tls_packet_uint16(packet, sni_len + 5); + // sni len + tls_packet_uint16(packet, sni_len + 3); + // sni type + tls_packet_uint8(packet, 0); + // sni host len + tls_packet_uint16(packet, sni_len); + tls_packet_append(packet, (unsigned char *)context->sni, sni_len); + } +#ifdef TLS_FORWARD_SECRECY +#ifdef TLS_CLIENT_ECDHE + // supported groups + tls_packet_uint16(packet, 0x0A); + tls_packet_uint16(packet, 8); + // 3 curves x 2 bytes + tls_packet_uint16(packet, 6); + tls_packet_uint16(packet, secp256r1.iana); + tls_packet_uint16(packet, secp384r1.iana); +#ifdef TLS_CURVE25519 + tls_packet_uint16(packet, x25519.iana); +#else + tls_packet_uint16(packet, secp224r1.iana); +#endif +#endif +#endif + if (alpn_len) { + tls_packet_uint16(packet, 0x10); + tls_packet_uint16(packet, alpn_len + 2); + tls_packet_uint16(packet, alpn_len); + + for (i = 0; i < context->alpn_count;i++) { + if (context->alpn[i]) { + int len = strlen(context->alpn[i]); + if (len) { + tls_packet_uint8(packet, len); + tls_packet_append(packet, (unsigned char *)context->alpn[i], len); + } + } + } + } + + if (context->dtls == 4) { + tls_packet_uint16(packet, 0x0E); + tls_packet_uint16(packet, 5); + tls_packet_uint16(packet, 2); + tls_packet_uint16(packet, SRTP_AES128_CM_HMAC_SHA1_80); + tls_packet_uint8(packet, 0); + + /* tls_packet_uint16(packet, 0x0E); + tls_packet_uint16(packet, 11); + tls_packet_uint16(packet, 8); + tls_packet_uint16(packet, SRTP_AEAD_AES_128_GCM); + tls_packet_uint16(packet, SRTP_AEAD_AES_256_GCM); + tls_packet_uint16(packet, SRTP_AES128_CM_HMAC_SHA1_80); + tls_packet_uint16(packet, SRTP_AES128_CM_HMAC_SHA1_32); + tls_packet_uint8(packet, 0); */ + } + + } + } +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + // supported versions + tls_packet_uint16(packet, 0x2B); + if (context->is_server) { + tls_packet_uint16(packet, 2); + if (context->version == TLS_V13) + tls_packet_uint16(packet, context->tls13_version ? context->tls13_version : TLS_V13); + else + tls_packet_uint16(packet, context->version); + } else { + tls_packet_uint16(packet, 5); + tls_packet_uint8(packet, 4); + tls_packet_uint16(packet, TLS_V13); + tls_packet_uint16(packet, 0x7F1C); + } + if (context->connection_status == 4) { + // fallback to the mandatory secp256r1 + tls_packet_uint16(packet, 0x33); + tls_packet_uint16(packet, 2); + tls_packet_uint16(packet, (unsigned short)secp256r1.iana); + } + if (((shared_key_short) && (selected_group)) || (!context->is_server)) { + // key share + tls_packet_uint16(packet, 0x33); + if (context->is_server) { + tls_packet_uint16(packet, shared_key_short + 4); + tls_packet_uint16(packet, (unsigned short)selected_group); + tls_packet_uint16(packet, shared_key_short); + tls_packet_append(packet, (unsigned char *)shared_key, shared_key_short); + } else { +#ifdef TLS_CURVE25519 + // make key + shared_key_short = 32; + tls_packet_uint16(packet, shared_key_short + 6); + tls_packet_uint16(packet, shared_key_short + 4); + + TLS_FREE(context->client_secret); + context->client_secret = (unsigned char *)TLS_MALLOC(32); + if (!context->client_secret) { + DEBUG_PRINT("ERROR IN TLS_MALLOC"); + TLS_FREE(packet); + return NULL; + + } + + static const unsigned char basepoint[32] = {9}; + + tls_random(context->client_secret, 32); + + context->client_secret[0] &= 248; + context->client_secret[31] &= 127; + context->client_secret[31] |= 64; + + curve25519(shared_key, context->client_secret, basepoint); + + tls_packet_uint16(packet, (unsigned short)x25519.iana); + tls_packet_uint16(packet, shared_key_short); + tls_packet_append(packet, (unsigned char *)shared_key, shared_key_short); +#else + // make key + shared_key_short = 65; + tls_packet_uint16(packet, shared_key_short + 6); + tls_packet_uint16(packet, shared_key_short + 4); + + _private_tls_ecc_dhe_create(context); + ltc_ecc_set_type *dp = (ltc_ecc_set_type *)&secp256r1.dp; + + if (ecc_make_key_ex(NULL, find_prng("sprng"), context->ecc_dhe, dp)) { + TLS_FREE(context->ecc_dhe); + context->ecc_dhe = NULL; + DEBUG_PRINT("Error generating ECC key\n"); + TLS_FREE(packet); + return NULL; + } + unsigned char out[TLS_MAX_RSA_KEY]; + unsigned long out_len = shared_key_short; + if (ecc_ansi_x963_export(context->ecc_dhe, out, &out_len)) { + DEBUG_PRINT("Error exporting ECC key\n"); + TLS_FREE(packet); + return NULL; + } + + tls_packet_uint16(packet, (unsigned short)secp256r1.iana); + tls_packet_uint16(packet, out_len); + tls_packet_append(packet, (unsigned char *)out, shared_key_short); +#endif + } + } + } +#endif + if ((context->version == TLS_V12) || (context->version == TLS_V13) || (context->version == DTLS_V13)) { + if (!context->is_server) { + // signature algorithms + tls_packet_uint16(packet, 0x0D); // type + tls_packet_uint16(packet, 2 + 2 * TLS_HASH_ALGO_NUMBER * TLS_SIGN_ALGO_NUMBER); // length + tls_packet_uint16(packet, 2 * TLS_HASH_ALGO_NUMBER * TLS_SIGN_ALGO_NUMBER); // actual length of the list and items themselves further + for (TLSHashAlgorithm hash = md5; !(hash > sha512); ++hash) { + for (TLSSignatureAlgorithm sign = rsa; !(sign > ecdsa); ++sign) { + tls_packet_uint16(packet, ((uint16_t)(hash) << 8) | (sign & 0xFF)); + } + } + } + } + + if ((!packet->broken) && (packet->buf)) { + int remaining = packet->len - start_len; + int payload_pos = 6; + if (context->dtls) + payload_pos = 14; + packet->buf[payload_pos] = remaining / 0x10000; + remaining %= 0x10000; + packet->buf[payload_pos + 1] = remaining / 0x100; + remaining %= 0x100; + packet->buf[payload_pos + 2] = remaining; + if (context->dtls) { + _private_dtls_handshake_copyframesize(packet); + context->dtls_seq ++; + } + } + tls_packet_update(packet); + } + return packet; +} + +struct TLSPacket *tls_certificate_request(struct TLSContext *context) { + if ((!context) || (!context->is_server)) + return NULL; + + unsigned short packet_version = context->version; + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, packet_version, 0); + if (packet) { + // certificate request + tls_packet_uint8(packet, 0x0D); + unsigned char dummy[3]; + tls_packet_append(packet, dummy, 3); + if (context->dtls) + _private_dtls_handshake_data(context, packet, 0); + int start_len = packet->len; +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + // certificate request context + tls_packet_uint8(packet, 0); + // extensions + tls_packet_uint16(packet, 18); + // signature algorithms + tls_packet_uint16(packet, 0x0D); + tls_packet_uint16(packet, 14); + tls_packet_uint16(packet, 12); + // rsa_pkcs1_sha256 + // tls_packet_uint16(packet, 0x0401); + // rsa_pkcs1_sha384 + // tls_packet_uint16(packet, 0x0501); + // rsa_pkcs1_sha512 + // tls_packet_uint16(packet, 0x0601); + + // ecdsa_secp256r1_sha256 + tls_packet_uint16(packet, 0x0403); + // ecdsa_secp384r1_sha384 + tls_packet_uint16(packet, 0x0503); + // ecdsa_secp521r1_sha512 + tls_packet_uint16(packet, 0x0604); + // rsa_pss_rsae_sha256 + tls_packet_uint16(packet, 0x0804); + // rsa_pss_rsae_sha384 + tls_packet_uint16(packet, 0x0805); + // rsa_pss_rsae_sha512 + tls_packet_uint16(packet, 0x0806); + } else +#endif + { +#ifdef TLS_ECDSA_SUPPORTED + tls_packet_uint8(packet, 2); + tls_packet_uint8(packet, rsa_sign); + tls_packet_uint8(packet, ecdsa_sign); +#else + tls_packet_uint8(packet, 1); + tls_packet_uint8(packet, rsa_sign); +#endif + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) { + // 10 pairs of 2 bytes +#ifdef TLS_ECDSA_SUPPORTED + tls_packet_uint16(packet, 14); + // ecdsa_secp256r1_sha256 + tls_packet_uint16(packet, 0x0403); + // // ecdsa_secp384r1_sha384 + tls_packet_uint16(packet, 0x0503); +#else + tls_packet_uint16(packet, 10); +#endif + tls_packet_uint8(packet, sha256); + tls_packet_uint8(packet, rsa); + tls_packet_uint8(packet, sha1); + tls_packet_uint8(packet, rsa); + tls_packet_uint8(packet, sha384); + tls_packet_uint8(packet, rsa); + tls_packet_uint8(packet, sha512); + tls_packet_uint8(packet, rsa); + tls_packet_uint8(packet, md5); + tls_packet_uint8(packet, rsa); + } + // no DistinguishedName yet + tls_packet_uint16(packet, 0); + } + if (!packet->broken) { + int remaining = packet->len - start_len; + int payload_pos = 6; + if (context->dtls) + payload_pos = 14; + packet->buf[payload_pos] = remaining / 0x10000; + remaining %= 0x10000; + packet->buf[payload_pos + 1] = remaining / 0x100; + remaining %= 0x100; + packet->buf[payload_pos + 2] = remaining; + + if (context->dtls) { + _private_dtls_handshake_copyframesize(packet); + context->dtls_seq++; + } + } + tls_packet_update(packet); + } + return packet; +} + +int _private_dtls_build_cookie(struct TLSContext *context) { + if ((!context->dtls_cookie) || (!context->dtls_cookie_len)) { + context->dtls_cookie = (unsigned char *)TLS_MALLOC(DTLS_COOKIE_SIZE); + if (!context->dtls_cookie) + return 0; + +#ifdef WITH_RANDOM_DLTS_COOKIE + if (!tls_random(context->dtls_cookie, DTLS_COOKIE_SIZE)) { + TLS_FREE(context->dtls_cookie); + context->dtls_cookie = NULL; + return 0; + } + context->dtls_cookie_len = DTLS_COOKIE_SIZE; +#else + hmac_state hmac; + hmac_init(&hmac, find_hash("sha256"), dtls_secret, sizeof(dtls_secret)); + hmac_process(&hmac, context->remote_random, TLS_CLIENT_RANDOM_SIZE); + + unsigned long out_size = DTLS_COOKIE_SIZE; + hmac_done(&hmac, context->dtls_cookie, &out_size); + + context->dtls_cookie_len = out_size; +#endif + } + return 1; +} + +struct TLSPacket *tls_build_verify_request(struct TLSContext *context) { + if ((!context->is_server) || (!context->dtls)) + return NULL; + + if ((!context->dtls_cookie) || (!context->dtls_cookie_len)) { + if (!_private_dtls_build_cookie(context)) + return NULL; + } + + unsigned short packet_version = context->version; + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, packet_version, 0); + if (packet) { + // verify request + tls_packet_uint8(packet, 0x03); + // 24-bit length + tls_packet_uint24(packet, context->dtls_cookie_len + 3); + // 16-bit message_sequence + tls_packet_uint16(packet, 0); + // 24-bit fragment_offset + tls_packet_uint24(packet, 0); + // 24-bit fragment_length + tls_packet_uint24(packet, context->dtls_cookie_len + 3); + // server_version + tls_packet_uint16(packet, context->version); + tls_packet_uint8(packet, context->dtls_cookie_len); + tls_packet_append(packet, context->dtls_cookie, context->dtls_cookie_len); + tls_packet_update(packet); + } + return packet; +} + +int _private_dtls_check_packet(struct TLSContext *context, const unsigned char *buf, int buf_len) { + CHECK_SIZE(11, buf_len, TLS_NEED_MORE_DATA) + + unsigned int bytes_to_follow = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + // not used: unsigned short message_seq = ntohs(*(unsigned short *)&buf[3]); + unsigned int fragment_offset = buf[5] * 0x10000 + buf[6] * 0x100 + buf[7]; + unsigned int fragment_length = buf[8] * 0x10000 + buf[9] * 0x100 + buf[10]; + + if ((fragment_offset) || (fragment_length != bytes_to_follow)) { + if ((context->dtls_data->fragment) && (context->dtls_data->fragment->written == bytes_to_follow)) + return bytes_to_follow; + + return TLS_FEATURE_NOT_SUPPORTED; + } + return bytes_to_follow; +} + +void _private_dtls_reset(struct TLSContext *context) { + context->dtls_epoch_local = 0; + context->dtls_epoch_remote = 0; + // context->local_sequence_number = 1; + context->dtls_seq = 0; + _private_tls_destroy_hash(context); + context->connection_status = 0; +} + +int tls_parse_verify_request(struct TLSContext *context, const unsigned char *buf, int buf_len, unsigned int *write_packets) { + *write_packets = 0; + if ((context->connection_status != 0) || (!context->dtls)) { + DEBUG_PRINT("UNEXPECTED VERIFY REQUEST MESSAGE\n"); + return TLS_UNEXPECTED_MESSAGE; + } + int res = 11; + int bytes_to_follow = _private_dtls_check_packet(context, buf, buf_len); + if (bytes_to_follow < 0) + return bytes_to_follow; + + CHECK_SIZE(bytes_to_follow, buf_len - res, TLS_NEED_MORE_DATA) + // not used: unsigned short version = ntohs(*(unsigned short *)&buf[res]); + res += 2; + unsigned char len = buf[res]; + res++; + TLS_FREE(context->dtls_cookie); + context->dtls_cookie_len = 0; + if (len) { + CHECK_SIZE(len, buf_len - res, TLS_NEED_MORE_DATA) + context->dtls_cookie = (unsigned char *)TLS_MALLOC(len); + if (!context->dtls_cookie) + return TLS_NO_MEMORY; + context->dtls_cookie_len = len; + memcpy(context->dtls_cookie, &buf[res], len); + res += len; + *write_packets = 4; + } + + // reset context + _private_dtls_reset(context); + return res; +} + +void _private_dtls_reset_cookie(struct TLSContext *context) { + TLS_FREE(context->dtls_cookie); + context->dtls_cookie = NULL; + context->dtls_cookie_len = 0; +} + +#ifdef WITH_TLS_13 +int _private_tls_parse_key_share(struct TLSContext *context, const unsigned char *buf, int buf_len) { + int i = 0; + struct ECCCurveParameters *curve = 0; + DHKey *dhkey = 0; + int dhe_key_size = 0; + const unsigned char *buffer = NULL; + unsigned char *out2; + unsigned long out_size; + unsigned short key_size = 0; + while (buf_len >= 4) { + unsigned short named_group = ntohs(*(unsigned short *)&buf[i]); + i += 2; + buf_len -= 2; + + key_size = ntohs(*(unsigned short *)&buf[i]); + i += 2; + buf_len -= 2; + + if (key_size > buf_len) + return TLS_BROKEN_PACKET; + + switch (named_group) { + case 0x0017: + curve = &secp256r1; + buffer = &buf[i]; + DEBUG_PRINT("KEY SHARE => secp256r1\n"); + buf_len = 0; + continue; + case 0x0018: + // secp384r1 + curve = &secp384r1; + buffer = &buf[i]; + DEBUG_PRINT("KEY SHARE => secp384r1\n"); + buf_len = 0; + continue; + case 0x0019: + // secp521r1 + break; + case 0x001D: + // x25519 +#ifdef TLS_CURVE25519 + if (key_size != 32) { + DEBUG_PRINT("INVALID x25519 KEY SIZE (%i)\n", key_size); + continue; + } + curve = &x25519; + buffer = &buf[i]; + DEBUG_PRINT("KEY SHARE => x25519\n"); + buf_len = 0; + continue; +#endif + break; + + case 0x001E: + // x448 + break; + case 0x0100: + dhkey = &ffdhe2048; + dhe_key_size = 2048; + break; + case 0x0101: + dhkey = &ffdhe3072; + dhe_key_size = 3072; + break; + case 0x0102: + dhkey = &ffdhe4096; + dhe_key_size = 4096; + break; + case 0x0103: + dhkey = &ffdhe6144; + dhe_key_size = 6144; + break; + case 0x0104: + dhkey = &ffdhe8192; + dhe_key_size = 8192; + break; + } + i += key_size; + buf_len -= key_size; + + if (!context->is_server) + break; + } + tls_init(); + if (curve) { + context->curve = curve; +#ifdef TLS_CURVE25519 + if (curve == &x25519) { + if ((context->is_server) && (!tls_random(context->local_random, TLS_SERVER_RANDOM_SIZE))) + return TLS_GENERIC_ERROR; + unsigned char secret[32]; + static const unsigned char basepoint[32] = {9}; + + if ((context->is_server) || (!context->client_secret)) { + tls_random(secret, 32); + + secret[0] &= 248; + secret[31] &= 127; + secret[31] |= 64; + + // use finished key to store public key + TLS_FREE(context->finished_key); + context->finished_key = (unsigned char *)TLS_MALLOC(32); + if (!context->finished_key) + return TLS_GENERIC_ERROR; + + curve25519(context->finished_key, secret, basepoint); + + TLS_FREE(context->premaster_key); + context->premaster_key = (unsigned char *)TLS_MALLOC(32); + if (!context->premaster_key) + return TLS_GENERIC_ERROR; + + curve25519(context->premaster_key, secret, buffer); + context->premaster_key_len = 32; + } else { + TLS_FREE(context->premaster_key); + context->premaster_key = (unsigned char *)TLS_MALLOC(32); + if (!context->premaster_key) + return TLS_GENERIC_ERROR; + + curve25519(context->premaster_key, context->client_secret, buffer); + context->premaster_key_len = 32; + + TLS_FREE(context->client_secret); + context->client_secret = NULL; + } + DEBUG_DUMP_HEX_LABEL("x25519 KEY", context->premaster_key, context->premaster_key_len); + + return 0; + } +#endif + if (context->is_server) { + _private_tls_ecc_dhe_create(context); + if (ecc_make_key_ex(NULL, find_prng("sprng"), context->ecc_dhe, (ltc_ecc_set_type *)&context->curve->dp)) { + TLS_FREE(context->ecc_dhe); + context->ecc_dhe = NULL; + DEBUG_PRINT("Error generating ECC DHE key\n"); + return TLS_GENERIC_ERROR; + } + } + + ltc_ecc_set_type *dp = (ltc_ecc_set_type *)&context->curve->dp; + + if ((context->is_server) && (!tls_random(context->local_random, TLS_SERVER_RANDOM_SIZE))) + return TLS_GENERIC_ERROR; + + ecc_key client_key; + memset(&client_key, 0, sizeof(client_key)); + if (ecc_ansi_x963_import_ex(buffer, key_size, &client_key, dp)) { + DEBUG_PRINT("Error importing ECC DHE key\n"); + return TLS_GENERIC_ERROR; + } + out2 = (unsigned char *)TLS_MALLOC(key_size); + out_size = key_size; + + int err = ecc_shared_secret(context->ecc_dhe, &client_key, out2, &out_size); + ecc_free(&client_key); + + if (err) { + DEBUG_PRINT("ECC DHE DECRYPT ERROR %i\n", err); + TLS_FREE(out2); + return TLS_GENERIC_ERROR; + } + DEBUG_PRINT("OUT_SIZE: %lu\n", out_size); + DEBUG_DUMP_HEX_LABEL("ECC DHE", out2, out_size); + + TLS_FREE(context->premaster_key); + context->premaster_key = out2; + context->premaster_key_len = out_size; + return 0; + } else + if ((dhkey) && (buffer)) { + _private_tls_dhe_create(context); + if (!tls_random(context->local_random, TLS_SERVER_RANDOM_SIZE)) + return TLS_GENERIC_ERROR; + if (_private_tls_dh_make_key(dhe_key_size / 8, context->dhe, (const char *)dhkey->p, (const char *)dhkey->g, 0, 0)) { + TLS_FREE(context->dhe); + context->dhe = NULL; + DEBUG_PRINT("Error generating DHE key\n"); + return TLS_GENERIC_ERROR; + } + + unsigned int dhe_out_size; + out2 = _private_tls_decrypt_dhe(context, buffer, key_size, &dhe_out_size, 0); + if (!out2) { + DEBUG_PRINT("Error generating DHE shared key\n"); + return TLS_GENERIC_ERROR; + } + + TLS_FREE(context->premaster_key); + context->premaster_key = out2; + context->premaster_key_len = dhe_out_size; + if (context->dhe) + context->dhe->iana = dhkey->iana; + return 0; + } + DEBUG_PRINT("NO COMMON KEY SHARE SUPPORTED\n"); + return TLS_NO_COMMON_CIPHER; +} +#endif + +int tls_parse_hello(struct TLSContext *context, const unsigned char *buf, int buf_len, unsigned int *write_packets, unsigned int *dtls_verified) { + *write_packets = 0; + *dtls_verified = 0; + if ((context->connection_status != 0) && (context->connection_status != 4)) { + // ignore multiple hello on dtls + if (context->dtls) { + DEBUG_PRINT("RETRANSMITTED HELLO MESSAGE RECEIVED\n"); + return 1; + } + DEBUG_PRINT("UNEXPECTED HELLO MESSAGE\n"); + return TLS_UNEXPECTED_MESSAGE; + } + + int res = 0; + int downgraded = 0; + int hello_min_size = context->dtls ? TLS_CLIENT_HELLO_MINSIZE + 8 : TLS_CLIENT_HELLO_MINSIZE; + CHECK_SIZE(hello_min_size, buf_len, TLS_NEED_MORE_DATA) + // big endian + unsigned int bytes_to_follow = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + res += 3; + if (context->dtls) { + int dtls_check = _private_dtls_check_packet(context, buf, buf_len); + if (dtls_check < 0) + return dtls_check; + // 16 bit message seq + 24 bit fragment offset + 24 bit fragment length + res += 8; + } + CHECK_SIZE(bytes_to_follow, buf_len - res, TLS_NEED_MORE_DATA) + + CHECK_SIZE(2, buf_len - res, TLS_NEED_MORE_DATA) + unsigned short version = ntohs(*(unsigned short *)&buf[res]); + unsigned short cipher = 0; + + res += 2; + VERSION_SUPPORTED(version, TLS_NOT_SAFE) + DEBUG_PRINT("VERSION REQUIRED BY REMOTE %x, VERSION NOW %x\n", (int)version, (int)context->version); +#ifdef TLS_LEGACY_SUPPORT + // when no legacy support, don't downgrade +#ifndef TLS_FORCE_LOCAL_VERSION + // downgrade ? + if (context->dtls) { + // for dlts, newer version has lower id (1.0 = FEFF, 1.2 = FEFD) + if (context->version < version) + downgraded = 1; + } else { + if (context->version > version) + downgraded = 1; + } + if (downgraded) { + context->version = version; + if (!context->is_server) + _private_tls_change_hash_type(context); + } +#endif +#endif + memcpy(context->remote_random, &buf[res], TLS_CLIENT_RANDOM_SIZE); + res += TLS_CLIENT_RANDOM_SIZE; + + unsigned char session_len = buf[res++]; + CHECK_SIZE(session_len, buf_len - res, TLS_NEED_MORE_DATA) + if ((session_len) && (session_len <= TLS_MAX_SESSION_ID)) { + memcpy(context->session, &buf[res], session_len); + context->session_size = session_len; + DEBUG_DUMP_HEX_LABEL("REMOTE SESSION ID: ", context->session, context->session_size); + } else + if (!context->dtls) + context->session_size = 0; + res += session_len; + + const unsigned char *cipher_buffer = NULL; + unsigned short cipher_len = 0; + int scsv_set = 0; + if (context->is_server) { + if (context->dtls) { + CHECK_SIZE(1, buf_len - res, TLS_NEED_MORE_DATA) + unsigned char tls_cookie_len = buf[res++]; + if (tls_cookie_len) { + CHECK_SIZE(tls_cookie_len, buf_len - res, TLS_NEED_MORE_DATA) + if ((!context->dtls_cookie_len) || (!context->dtls_cookie)) + _private_dtls_build_cookie(context); + + if ((context->dtls_cookie_len != tls_cookie_len) || (!context->dtls_cookie)) { + *dtls_verified = 2; + // _private_dtls_reset_cookie(context); + DEBUG_PRINT("INVALID DTLS COOKIE\n"); + return TLS_BROKEN_PACKET; + } + if (memcmp(context->dtls_cookie, &buf[res], tls_cookie_len)) { + *dtls_verified = 3; + // _private_dtls_reset_cookie(context); + DEBUG_PRINT("MISMATCH DTLS COOKIE\n"); + return TLS_BROKEN_PACKET; + } + // _private_dtls_reset_cookie(context); + context->dtls_seq ++; + *dtls_verified = 1; + res += tls_cookie_len; + } else { + *write_packets = 2; + return buf_len; + } + } + CHECK_SIZE(2, buf_len - res, TLS_NEED_MORE_DATA) + cipher_len = ntohs(*(unsigned short *)&buf[res]); + res += 2; + CHECK_SIZE(cipher_len, buf_len - res, TLS_NEED_MORE_DATA) + // faster than cipher_len % 2 + if (cipher_len & 1) + return TLS_BROKEN_PACKET; + + cipher_buffer = &buf[res]; + res += cipher_len; + + CHECK_SIZE(1, buf_len - res, TLS_NEED_MORE_DATA) + unsigned char compression_list_size = buf[res++]; + CHECK_SIZE(compression_list_size, buf_len - res, TLS_NEED_MORE_DATA) + + // no compression support + res += compression_list_size; + } else { + CHECK_SIZE(2, buf_len - res, TLS_NEED_MORE_DATA) + cipher = ntohs(*(unsigned short *)&buf[res]); + res += 2; + context->cipher = cipher; +#ifndef WITH_TLS_13 + if (!tls_cipher_supported(context, cipher)) { + context->cipher = 0; + DEBUG_PRINT("NO CIPHER SUPPORTED\n"); + return TLS_NO_COMMON_CIPHER; + } + DEBUG_PRINT("CIPHER: %s\n", tls_cipher_name(context)); +#endif + CHECK_SIZE(1, buf_len - res, TLS_NEED_MORE_DATA) + unsigned char compression = buf[res++]; + if (compression != 0) { + DEBUG_PRINT("COMPRESSION NOT SUPPORTED\n"); + return TLS_COMPRESSION_NOT_SUPPORTED; + } + } + + if (res > 0) { + if (context->is_server) + *write_packets = 2; + if (context->connection_status != 4) + context->connection_status = 1; + } + + + if (res > 2) + res += 2; +#ifdef WITH_TLS_13 + const unsigned char *key_share = NULL; + unsigned short key_size = 0; +#endif + while (buf_len - res >= 4) { + // have extensions + unsigned short extension_type = ntohs(*(unsigned short *)&buf[res]); + res += 2; + unsigned short extension_len = ntohs(*(unsigned short *)&buf[res]); + res += 2; + DEBUG_PRINT("Extension: 0x0%x (%i), len: %i\n", (int)extension_type, (int)extension_type, (int)extension_len); + // SNI extension + CHECK_SIZE(extension_len, buf_len - res, TLS_NEED_MORE_DATA) + if (extension_type == 0x00) { + // unsigned short sni_len = ntohs(*(unsigned short *)&buf[res]); + // unsigned char sni_type = buf[res + 2]; + unsigned short sni_host_len = ntohs(*(unsigned short *)&buf[res + 3]); + CHECK_SIZE(sni_host_len, buf_len - res - 5, TLS_NEED_MORE_DATA) + if (sni_host_len) { + TLS_FREE(context->sni); + context->sni = (char *)TLS_MALLOC(sni_host_len + 1); + if (context->sni) { + memcpy(context->sni, &buf[res + 5], sni_host_len); + context->sni[sni_host_len] = 0; + DEBUG_PRINT("SNI HOST INDICATOR: [%s]\n", context->sni); + } + } + } else +#ifdef TLS_FORWARD_SECRECY + if (extension_type == 0x0A) { + // supported groups + if (buf_len - res > 2) { + unsigned short group_len = ntohs(*(unsigned short *)&buf[res]); + if (buf_len - res >= group_len + 2) { + DEBUG_DUMP_HEX_LABEL("SUPPORTED GROUPS", &buf[res + 2], group_len); + int i; + int selected = 0; + for (i = 0; i < group_len; i += 2) { + unsigned short iana_n = ntohs(*(unsigned short *)&buf[res + 2 + i]); + switch (iana_n) { + case 23: + context->curve = &secp256r1; + selected = 1; + break; + case 24: + context->curve = &secp384r1; + selected = 1; + break; +#ifdef WITH_TLS_13 + // needs different implementation + // case 29: + // context->curve = &x25519; + // selected = 1; + // break; +#endif + // do not use it anymore + // case 25: + // context->curve = &secp521r1; + // selected = 1; + // break; + } + if (selected) { + DEBUG_PRINT("SELECTED CURVE %s\n", context->curve->name); + break; + } + } + } + } + } else +#endif + if ((extension_type == 0x10) && (context->alpn) && (context->alpn_count)) { + if (buf_len - res > 2) { + unsigned short alpn_len = ntohs(*(unsigned short *)&buf[res]); + if ((alpn_len) && (alpn_len <= extension_len - 2)) { + unsigned char *alpn = (unsigned char *)&buf[res + 2]; + int alpn_pos = 0; + while (alpn_pos < alpn_len) { + unsigned char alpn_size = alpn[alpn_pos++]; + if (alpn_size + alpn_pos >= extension_len) + break; + if ((alpn_size) && (tls_alpn_contains(context, (char *)&alpn[alpn_pos], alpn_size))) { + TLS_FREE(context->negotiated_alpn); + context->negotiated_alpn = (char *)TLS_MALLOC(alpn_size + 1); + if (context->negotiated_alpn) { + memcpy(context->negotiated_alpn, &alpn[alpn_pos], alpn_size); + context->negotiated_alpn[alpn_size] = 0; + DEBUG_PRINT("NEGOTIATED ALPN: %s\n", context->negotiated_alpn); + } + break; + } + alpn_pos += alpn_size; + // ServerHello contains just one alpn + if (!context->is_server) + break; + } + } + } + } else + if (extension_type == 0x0D) { + // supported signatures + DEBUG_DUMP_HEX_LABEL("SUPPORTED SIGNATURES", &buf[res], extension_len); + } else + if (extension_type == 0x0B) { + // supported point formats + DEBUG_DUMP_HEX_LABEL("SUPPORTED POINT FORMATS", &buf[res], extension_len); + } else + if ((extension_type == 0x0E) && (context->dtls)) { + // use_srtp + DEBUG_DUMP_HEX_LABEL("USE SRTP", &buf[res], extension_len); + context->dtls = 4; + } else + if ((extension_type == 0x17) && (context->dtls)) { + // extended_master_secret + DEBUG_PRINT("EXTENDED MASTER SECRET"); +#ifdef TLS_DTLS_EXTENDED_MASTER_SECRET + if (context->dtls_data) + context->dtls_data->extended_master_secret = 1; +#endif + } +#ifdef WITH_TLS_13 + else + if (extension_type == 0x2B) { + // supported versions + if ((context->is_server) && (buf[res] == extension_len - 1)) { + if (extension_len > 2) { + DEBUG_DUMP_HEX_LABEL("SUPPORTED VERSIONS", &buf[res], extension_len); + int i; + int limit = (int)buf[res]; + if (limit == extension_len - 1) { + for (i = 1; i < limit; i += 2) { + if ((ntohs(*(unsigned short *)&buf[res + i]) == TLS_V13) || (ntohs(*(unsigned short *)&buf[res + i]) == 0x7F1C)) { + context->version = TLS_V13; + context->tls13_version = ntohs(*(unsigned short *)&buf[res + i]); + DEBUG_PRINT("TLS 1.3 SUPPORTED\n"); + break; + } + } + } + } + } else + if ((!context->is_server) && (extension_len == 2)) { + if ((ntohs(*(unsigned short *)&buf[res]) == TLS_V13) || (ntohs(*(unsigned short *)&buf[res]) == 0x7F1C)) { + context->version = TLS_V13; + context->tls13_version = ntohs(*(unsigned short *)&buf[res]); + DEBUG_PRINT("TLS 1.3 SUPPORTED\n"); + } + } + } else + if (extension_type == 0x2A) { + // early data + DEBUG_DUMP_HEX_LABEL("EXTENSION, EARLY DATA", &buf[res], extension_len); + } else + if (extension_type == 0x29) { + // pre shared key + DEBUG_DUMP_HEX_LABEL("EXTENSION, PRE SHARED KEY", &buf[res], extension_len); + } else + if (extension_type == 0x33) { + // key share + if (context->is_server) { + key_size = ntohs(*(unsigned short *)&buf[res]); + if ((context->is_server) && (key_size > extension_len - 2)) { + DEBUG_PRINT("BROKEN KEY SHARE\n"); + return TLS_BROKEN_PACKET; + } + } else { + key_size = extension_len; + } + DEBUG_DUMP_HEX_LABEL("EXTENSION, KEY SHARE", &buf[res], extension_len); + if (context->is_server) + key_share = &buf[res + 2]; + else + key_share = &buf[res]; + } else + if (extension_type == 0x0D) { + // signature algorithms + DEBUG_DUMP_HEX_LABEL("EXTENSION, SIGNATURE ALGORITHMS", &buf[res], extension_len); + } else + if (extension_type == 0x2D) { + // psk key exchange modes + DEBUG_DUMP_HEX_LABEL("EXTENSION, PSK KEY EXCHANGE MODES", &buf[res], extension_len); + } +#endif + res += extension_len; + } + if (buf_len != res) + return TLS_NEED_MORE_DATA; + if ((context->is_server) && (cipher_buffer) && (cipher_len)) { + int cipher = tls_choose_cipher(context, cipher_buffer, cipher_len, &scsv_set); + if (cipher < 0) { + DEBUG_PRINT("NO COMMON CIPHERS\n"); + return cipher; + } + if ((downgraded) && (scsv_set)) { + DEBUG_PRINT("NO DOWNGRADE (SCSV SET)\n"); + _private_tls_write_packet(tls_build_alert(context, 1, inappropriate_fallback)); + context->critical_error = 1; + return TLS_NOT_SAFE; + } + context->cipher = cipher; + } +#ifdef WITH_TLS_13 + if (!context->is_server) { + if (!tls_cipher_supported(context, cipher)) { + context->cipher = 0; + DEBUG_PRINT("NO CIPHER SUPPORTED\n"); + return TLS_NO_COMMON_CIPHER; + } + DEBUG_PRINT("CIPHER: %s\n", tls_cipher_name(context)); + } + + if ((key_share) && (key_size) && ((context->version == TLS_V13) || (context->version == DTLS_V13))) { + int key_share_err = _private_tls_parse_key_share(context, key_share, key_size); + if (key_share_err) { + // request hello retry + if (context->connection_status != 4) { + *write_packets = 5; + context->hs_messages[1] = 0; + context->connection_status = 4; + return res; + } else + return key_share_err; + } + // we have key share + if (context->is_server) + context->connection_status = 3; + else + context->connection_status = 2; + } +#endif + return res; +} + +int tls_parse_certificate(struct TLSContext *context, const unsigned char *buf, int buf_len, int is_client) { + int res = 0; + CHECK_SIZE(3, buf_len, TLS_NEED_MORE_DATA) + unsigned int size_of_all_certificates = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + + if (size_of_all_certificates <= 4) + return 3 + size_of_all_certificates; + res += 3; + if (context->dtls) { + int dtls_check = _private_dtls_check_packet(context, buf, buf_len); + if (dtls_check < 0) + return dtls_check; + res += 8; + } +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + int context_size = buf[res]; + res++; + // must be 0 + if (context_size) + res += context_size; + size_of_all_certificates --; + } +#endif + + CHECK_SIZE(size_of_all_certificates, buf_len - res, TLS_NEED_MORE_DATA); + int size = size_of_all_certificates; + + int idx = 0; + int valid_certificate = 0; + while (size > 0) { + idx++; + CHECK_SIZE(3, buf_len - res, TLS_NEED_MORE_DATA); + unsigned int certificate_size = buf[res] * 0x10000 + buf[res + 1] * 0x100 + buf[res + 2]; + res += 3; + CHECK_SIZE(certificate_size, buf_len - res, TLS_NEED_MORE_DATA) + // load chain + int certificates_in_chain = 0; + int res2 = res; + unsigned int remaining = certificate_size; + do { + if (remaining <= 3) + break; + certificates_in_chain++; + unsigned int certificate_size2 = buf[res2] * 0x10000 + buf[res2 + 1] * 0x100 + buf[res2 + 2]; + res2 += 3; + remaining -= 3; + if (certificate_size2 > remaining) { + DEBUG_PRINT("Invalid certificate size (%i from %i bytes remaining)\n", certificate_size2, remaining); + break; + } + remaining -= certificate_size2; + + struct TLSCertificate *cert = asn1_parse(context, &buf[res2], certificate_size2, is_client); + if (cert) { + if (certificate_size2) { + cert->bytes = (unsigned char *)TLS_MALLOC(certificate_size2); + if (cert->bytes) { + cert->len = certificate_size2; + memcpy(cert->bytes, &buf[res2], certificate_size2); + } + } + if ((context->dtls_data) && (context->dtls_data->remote_fingerprint)) { + unsigned char hash[32]; + + hash_state state; + + sha256_init(&state); + sha256_process(&state, cert->bytes, cert->len); + sha256_done(&state, hash); + + int i; + char buffer_data[100]; + char *buffer = buffer_data; + int buf_len = sizeof(buffer_data); + buffer[0] = 0; + for (i = 0; i < 32; i++) { + if (buf_len <= 1) + break; + if (i) { + snprintf(buffer, buf_len, ":"); + buffer ++; + buf_len --; + } + if (buf_len <= 2) + break; + snprintf(buffer, buf_len, "%02X", (unsigned int)hash[i]); + buffer += 2; + buf_len -= 2; + } + if (strcmp(buffer_data, context->dtls_data->remote_fingerprint)) { + DEBUG_PRINT("PEER CERTIFICATE FINGERPRINT VALIDATION FAILED, computed %s, expected %s\n", buffer_data, context->dtls_data->remote_fingerprint); + return TLS_UNSUPPORTED_CERTIFICATE; + } + } + // valid certificate + if (is_client) { + valid_certificate = 1; + context->client_certificates = (struct TLSCertificate **)TLS_REALLOC(context->client_certificates, (context->client_certificates_count + 1) * sizeof(struct TLSCertificate *)); + context->client_certificates[context->client_certificates_count] = cert; + context->client_certificates_count++; + } else { + context->certificates = (struct TLSCertificate **)TLS_REALLOC(context->certificates, (context->certificates_count + 1) * sizeof(struct TLSCertificate *)); + context->certificates[context->certificates_count] = cert; + context->certificates_count++; + if ((cert->pk) || (cert->priv)) + valid_certificate = 1; + else + if (!context->is_server) + valid_certificate = 1; + } + } + res2 += certificate_size2; +#ifdef WITH_TLS_13 + // extension + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + if (remaining >= 2) { + // ignore extensions + remaining -= 2; + unsigned short size = ntohs(*(unsigned short *)&buf[res2]); + if ((size) && (size >= remaining)) { + res2 += size; + remaining -= size; + } + res2 += 2; + } + } +#endif + } while (remaining > 0); + if (remaining) { + DEBUG_PRINT("Extra %i bytes after certificate\n", remaining); + } + size -= certificate_size + 3; + res += certificate_size; + } + if (!valid_certificate) + return TLS_UNSUPPORTED_CERTIFICATE; + if (res != buf_len) { + DEBUG_PRINT("Warning: %i bytes read from %i byte buffer\n", (int)res, (int)buf_len); + } + return res; +} + +int _private_tls_parse_dh(const unsigned char *buf, int buf_len, const unsigned char **out, int *out_size) { + int res = 0; + *out = NULL; + *out_size = 0; + CHECK_SIZE(2, buf_len, TLS_NEED_MORE_DATA) + unsigned short size = ntohs(*(unsigned short *)buf); + res += 2; + CHECK_SIZE(size, buf_len - res, TLS_NEED_MORE_DATA) + DEBUG_DUMP_HEX(&buf[res], size); + *out = &buf[res]; + *out_size = size; + res += size; + return res; +} + +int _private_tls_parse_random(struct TLSContext *context, const unsigned char *buf, int buf_len) { + int res = 0; + int ephemeral = tls_cipher_is_ephemeral(context); + unsigned short size; + if (ephemeral == 2) { + CHECK_SIZE(1, buf_len, TLS_NEED_MORE_DATA) + size = buf[0]; + res += 1; + } else { + CHECK_SIZE(2, buf_len, TLS_NEED_MORE_DATA) + size = ntohs(*(unsigned short *)buf); + res += 2; + } + + CHECK_SIZE(size, buf_len - res, TLS_NEED_MORE_DATA) + unsigned int out_len = 0; + unsigned char *random = NULL; + switch (ephemeral) { +#ifdef TLS_FORWARD_SECRECY + case 1: + random = _private_tls_decrypt_dhe(context, &buf[res], size, &out_len, !context->dtls); + break; + case 2: + random = _private_tls_decrypt_ecc_dhe(context, &buf[res], size, &out_len, !context->dtls); + break; +#endif + default: + random = _private_tls_decrypt_rsa(context, &buf[res], size, &out_len); + } + + if ((random) && (out_len > 2)) { + DEBUG_DUMP_HEX_LABEL("PRE MASTER KEY", random, out_len); + TLS_FREE(context->premaster_key); + context->premaster_key = random; + context->premaster_key_len = out_len; + _private_tls_compute_key(context, 48); + } else { + TLS_FREE(random); + return 0; + } + res += size; + return res; +} + +int _private_tls_build_random(struct TLSPacket *packet) { + int res = 0; + unsigned char rand_bytes[48]; + int bytes = 48; + if (!tls_random(rand_bytes, bytes)) + return TLS_GENERIC_ERROR; + + // max supported version + if (packet->context->is_server) + *(unsigned short *)rand_bytes = htons(packet->context->version); + else + if (packet->context->dtls) + *(unsigned short *)rand_bytes = htons(DTLS_V12); + else + *(unsigned short *)rand_bytes = htons(TLS_V12); + //DEBUG_DUMP_HEX_LABEL("PREMASTER KEY", rand_bytes, bytes); + + TLS_FREE(packet->context->premaster_key); + packet->context->premaster_key = (unsigned char *)TLS_MALLOC(bytes); + if (!packet->context->premaster_key) + return TLS_NO_MEMORY; + + packet->context->premaster_key_len = bytes; + memcpy(packet->context->premaster_key, rand_bytes, packet->context->premaster_key_len); + + unsigned int out_len; + unsigned char *random = _private_tls_encrypt_rsa(packet->context, packet->context->premaster_key, packet->context->premaster_key_len, &out_len); + + _private_tls_compute_key(packet->context, bytes); + if ((random) && (out_len > 2)) { + tls_packet_uint24(packet, out_len + 2); + if (packet->context->dtls) + _private_dtls_handshake_data(packet->context, packet, out_len + 2); + tls_packet_uint16(packet, out_len); + tls_packet_append(packet, random, out_len); + } else + res = TLS_GENERIC_ERROR; + TLS_FREE(random); + if (res) + return res; + + return out_len + 2; +} + +const unsigned char *_private_tls_parse_signature(struct TLSContext *context, const unsigned char *buf, int buf_len, int *hash_algorithm, int *sign_algorithm, int *sig_size, int *offset) { + int res = 0; + CHECK_SIZE(2, buf_len, NULL) + *hash_algorithm = _md5_sha1; + *sign_algorithm = rsa_sign; + *sig_size = 0; + if ((context->version == TLS_V12) || (context->version == DTLS_V12) || (context->version == TLS_V13) || (context->version == DTLS_V13)) { + *hash_algorithm = buf[res]; + res++; + *sign_algorithm = buf[res]; + res++; + } + unsigned short size = ntohs(*(unsigned short *)&buf[res]); + res += 2; + CHECK_SIZE(size, buf_len - res, NULL) + DEBUG_DUMP_HEX(&buf[res], size); + *sig_size = size; + *offset = res + size; + return &buf[res]; +} + +int tls_parse_server_key_exchange(struct TLSContext *context, const unsigned char *buf, int buf_len) { + int res = 0; + int dh_res = 0; + CHECK_SIZE(3, buf_len, TLS_NEED_MORE_DATA) + unsigned int size = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + res += 3; + if (context->dtls) { + int dtls_check = _private_dtls_check_packet(context, buf, buf_len); + if (dtls_check < 0) + return dtls_check; + res += 8; + } + const unsigned char *packet_ref = buf + res; + CHECK_SIZE(size, buf_len - res, TLS_NEED_MORE_DATA); + + if (!size) + return res; + + unsigned char has_ds_params = 0; + unsigned int key_size = 0; +#ifdef TLS_FORWARD_SECRECY + const struct ECCCurveParameters *curve = NULL; + const unsigned char *pk_key = NULL; + int ephemeral = tls_cipher_is_ephemeral(context); + if (ephemeral) { + if (ephemeral == 1) { + has_ds_params = 1; + } else { + if (buf[res++] != 3) { + // named curve + // any other method is not supported + return 0; + } + CHECK_SIZE(3, buf_len - res, TLS_NEED_MORE_DATA); + int iana_n = ntohs(*(unsigned short *)&buf[res]); + res += 2; + key_size = buf[res]; + res++; + CHECK_SIZE(key_size, buf_len - res, TLS_NEED_MORE_DATA); + DEBUG_PRINT("IANA CURVE NUMBER: %i\n", iana_n); + switch (iana_n) { + case 19: + curve = &secp192r1; + break; + case 20: + curve = &secp224k1; + break; + case 21: + curve = &secp224r1; + break; + case 22: + curve = &secp256k1; + break; + case 23: + curve = &secp256r1; + break; + case 24: + curve = &secp384r1; + break; + case 25: + curve = &secp521r1; + break; +#ifdef TLS_CURVE25519 + case 29: + curve = &x25519; + break; +#endif + default: + DEBUG_PRINT("UNSUPPORTED CURVE\n"); + return TLS_GENERIC_ERROR; + } + pk_key = &buf[res]; + res += key_size; + context->curve = curve; + } + } +#endif + const unsigned char *dh_p = NULL; + int dh_p_len = 0; + const unsigned char *dh_g = NULL; + int dh_g_len = 0; + const unsigned char *dh_Ys = NULL; + int dh_Ys_len = 0; + if (has_ds_params) { + DEBUG_PRINT(" dh_p: "); + dh_res = _private_tls_parse_dh(&buf[res], buf_len - res, &dh_p, &dh_p_len); + if (dh_res <= 0) + return TLS_BROKEN_PACKET; + res += dh_res; + DEBUG_PRINT("\n"); + + DEBUG_PRINT(" dh_q: "); + dh_res = _private_tls_parse_dh(&buf[res], buf_len - res, &dh_g, &dh_g_len); + if (dh_res <= 0) + return TLS_BROKEN_PACKET; + res += dh_res; + DEBUG_PRINT("\n"); + + DEBUG_PRINT(" dh_Ys: "); + dh_res = _private_tls_parse_dh(&buf[res], buf_len - res, &dh_Ys, &dh_Ys_len); + if (dh_res <= 0) + return TLS_BROKEN_PACKET; + res += dh_res; + DEBUG_PRINT("\n"); + } + int sign_size; + int hash_algorithm; + int sign_algorithm; + int packet_size = res - 3; + if (context->dtls) + packet_size -= 8; + int offset = 0; + DEBUG_PRINT(" SIGNATURE (%i/%i/%i): ", packet_size, dh_res, key_size); + const unsigned char *signature = _private_tls_parse_signature(context, &buf[res], buf_len - res, &hash_algorithm, &sign_algorithm, &sign_size, &offset); + DEBUG_PRINT("\n"); + if ((sign_size <= 0) || (!signature)) + return TLS_BROKEN_PACKET; + res += offset; + // check signature + unsigned int message_len = packet_size + TLS_CLIENT_RANDOM_SIZE + TLS_SERVER_RANDOM_SIZE; + unsigned char *message = (unsigned char *)TLS_MALLOC(message_len); + if (message) { + memcpy(message, context->local_random, TLS_CLIENT_RANDOM_SIZE); + memcpy(message + TLS_CLIENT_RANDOM_SIZE, context->remote_random, TLS_SERVER_RANDOM_SIZE); + memcpy(message + TLS_CLIENT_RANDOM_SIZE + TLS_SERVER_RANDOM_SIZE, packet_ref, packet_size); +#ifdef TLS_CLIENT_ECDSA + if (tls_is_ecdsa(context)) { + if (_private_tls_verify_ecdsa(context, hash_algorithm, signature, sign_size, message, message_len, NULL) != 1) { + // DEBUG_PRINT("ECC Server signature FAILED!\n"); + // TLS_FREE(message); + // return TLS_BROKEN_PACKET; + } + } else +#endif + { + if (_private_tls_verify_rsa(context, hash_algorithm, signature, sign_size, message, message_len) != 1) { + // DEBUG_PRINT("Server signature FAILED!\n"); + // TLS_FREE(message); + // return TLS_BROKEN_PACKET; + } + } + TLS_FREE(message); + } + + if (buf_len - res) { + DEBUG_PRINT("EXTRA %i BYTES AT THE END OF MESSAGE\n", buf_len - res); + DEBUG_DUMP_HEX(&buf[res], buf_len - res); + DEBUG_PRINT("\n"); + } +#ifdef TLS_FORWARD_SECRECY + if (ephemeral == 1) { + _private_tls_dhe_create(context); + DEBUG_DUMP_HEX_LABEL("DHP", dh_p, dh_p_len); + DEBUG_DUMP_HEX_LABEL("DHG", dh_g, dh_g_len); + int dhe_key_size = dh_p_len; + if (dh_g_len > dh_p_len) + dhe_key_size = dh_g_len; + if (_private_tls_dh_make_key(dhe_key_size, context->dhe, (const char *)dh_p, (const char *)dh_g, dh_p_len, dh_g_len)) { + DEBUG_PRINT("ERROR CREATING DHE KEY\n"); + TLS_FREE(context->dhe); + context->dhe = NULL; + return TLS_GENERIC_ERROR; + } + + unsigned int dh_key_size = 0; + unsigned char *key = _private_tls_decrypt_dhe(context, dh_Ys, dh_Ys_len, &dh_key_size, 0); + DEBUG_DUMP_HEX_LABEL("DH COMMON SECRET", key, dh_key_size); + if ((key) && (dh_key_size)) { + TLS_FREE(context->premaster_key); + context->premaster_key = key; + context->premaster_key_len = dh_key_size; + } + } else + if ((ephemeral == 2) && (curve) && (pk_key) && (key_size)) { +#ifdef TLS_CURVE25519 + if (curve == &x25519) { + if (key_size != 32) { + DEBUG_PRINT("INVALID X25519 PUBLIC SIZE"); + return TLS_GENERIC_ERROR; + } + + TLS_FREE(context->client_secret); + context->client_secret = (unsigned char *)TLS_MALLOC(32); + if (!context->client_secret) { + DEBUG_PRINT("ERROR IN TLS_MALLOC"); + return TLS_GENERIC_ERROR; + } + + tls_random(context->client_secret, 32); + + context->client_secret[0] &= 248; + context->client_secret[31] &= 127; + context->client_secret[31] |= 64; + + TLS_FREE(context->premaster_key); + context->premaster_key = (unsigned char *)TLS_MALLOC(32); + if (!context->premaster_key) + return TLS_GENERIC_ERROR; + + curve25519(context->premaster_key, context->client_secret, pk_key); + context->premaster_key_len = 32; + } else +#endif + { + tls_init(); + _private_tls_ecc_dhe_create(context); + + ltc_ecc_set_type *dp = (ltc_ecc_set_type *)&curve->dp; + if (ecc_make_key_ex(NULL, find_prng("sprng"), context->ecc_dhe, dp)) { + TLS_FREE(context->ecc_dhe); + context->ecc_dhe = NULL; + DEBUG_PRINT("Error generating ECC key\n"); + return TLS_GENERIC_ERROR; + } + + TLS_FREE(context->premaster_key); + context->premaster_key_len = 0; + + unsigned int out_len = 0; + context->premaster_key = _private_tls_decrypt_ecc_dhe(context, pk_key, key_size, &out_len, 0); + if (context->premaster_key) + context->premaster_key_len = out_len; + } + } +#endif + return res; +} + +int tls_parse_client_key_exchange(struct TLSContext *context, const unsigned char *buf, int buf_len) { + if ((context->connection_status != 1) && (!context->dtls)) { + DEBUG_PRINT("UNEXPECTED CLIENT KEY EXCHANGE MESSAGE (connections status: %i)\n", (int)context->connection_status); + return TLS_UNEXPECTED_MESSAGE; + } + + int res = 0; + int dh_res = 0; + CHECK_SIZE(3, buf_len, TLS_NEED_MORE_DATA) + + unsigned int size = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + res += 3; + if (context->dtls) { + int dtls_check = _private_dtls_check_packet(context, buf, buf_len); + if (dtls_check < 0) + return dtls_check; + res += 8; + } + + CHECK_SIZE(size, buf_len - res, TLS_NEED_MORE_DATA); + + if (!size) + return res; + + dh_res = _private_tls_parse_random(context, &buf[res], size); + if (dh_res <= 0) { + DEBUG_PRINT("broken key\n"); + return TLS_BROKEN_PACKET; + } + DEBUG_PRINT("\n"); + + res += size; + context->connection_status = 2; + return res; +} + +int tls_parse_server_hello_done(struct TLSContext *context, const unsigned char *buf, int buf_len) { + int res = 0; + CHECK_SIZE(3, buf_len, TLS_NEED_MORE_DATA) + + unsigned int size = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + res += 3; + if (context->dtls) { + int dtls_check = _private_dtls_check_packet(context, buf, buf_len); + if (dtls_check < 0) + return dtls_check; + res += 8; + } + + CHECK_SIZE(size, buf_len - res, TLS_NEED_MORE_DATA); + + res += size; + return res; +} + +int tls_parse_finished(struct TLSContext *context, const unsigned char *buf, int buf_len, unsigned int *write_packets) { + if ((context->connection_status < 2) || (context->connection_status == 0xFF)) { + DEBUG_PRINT("UNEXPECTED FINISHED MESSAGE\n"); + return TLS_UNEXPECTED_MESSAGE; + } + + int res = 0; + *write_packets = 0; + CHECK_SIZE(3, buf_len, TLS_NEED_MORE_DATA) + + unsigned int size = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + res += 3; + if (context->dtls) { + int dtls_check = _private_dtls_check_packet(context, buf, buf_len); + if (dtls_check < 0) + return dtls_check; + res += 8; + } + + if (size < TLS_MIN_FINISHED_OPAQUE_LEN) { + DEBUG_PRINT("Invalid finished pachet size: %i\n", size); + return TLS_BROKEN_PACKET; + } + + CHECK_SIZE(size, buf_len - res, TLS_NEED_MORE_DATA); + + unsigned char hash[TLS_MAX_SHA_SIZE]; + unsigned int hash_len = _private_tls_get_hash(context, hash); + +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + unsigned char hash_out[TLS_MAX_SHA_SIZE]; + unsigned long out_size = TLS_MAX_SHA_SIZE; + if ((!context->remote_finished_key) || (!hash_len)) { + DEBUG_PRINT("NO FINISHED KEY COMPUTED OR NO HANDSHAKE HASH\n"); + return TLS_NOT_VERIFIED; + } + + DEBUG_DUMP_HEX_LABEL("HS HASH", hash, hash_len); + DEBUG_DUMP_HEX_LABEL("HS FINISH", context->finished_key, hash_len); + DEBUG_DUMP_HEX_LABEL("HS REMOTE FINISH", context->remote_finished_key, hash_len); + + out_size = hash_len; + hmac_state hmac; + hmac_init(&hmac, _private_tls_get_hash_idx(context), context->remote_finished_key, hash_len); + hmac_process(&hmac, hash, hash_len); + hmac_done(&hmac, hash_out, &out_size); + + if ((size != out_size) || (memcmp(hash_out, &buf[res], size))) { + DEBUG_PRINT("Finished validation error (sequence number, local: %i, remote: %i)\n", (int)context->local_sequence_number, (int)context->remote_sequence_number); + DEBUG_DUMP_HEX_LABEL("FINISHED OPAQUE", &buf[res], size); + DEBUG_DUMP_HEX_LABEL("VERIFY", hash_out, out_size); + return TLS_NOT_VERIFIED; + } + if (context->is_server) { + context->connection_status = 0xFF; + res += size; + _private_tls13_key(context, 0); + context->local_sequence_number = 0; + context->remote_sequence_number = 0; + return res; + } + } else +#endif + { + // verify + unsigned char *out = (unsigned char *)TLS_MALLOC(size); + if (!out) { + DEBUG_PRINT("Error in TLS_MALLOC (%i bytes)\n", (int)size); + return TLS_NO_MEMORY; + } + + // server verifies client's message + if (context->is_server) + _private_tls_prf(context, out, size, context->master_key, context->master_key_len, (unsigned char *)"client finished", 15, hash, hash_len, NULL, 0); + else + _private_tls_prf(context, out, size, context->master_key, context->master_key_len, (unsigned char *)"server finished", 15, hash, hash_len, NULL, 0); + + if (memcmp(out, &buf[res], size)) { + TLS_FREE(out); + DEBUG_PRINT("Finished validation error (sequence number, local: %i, remote: %i)\n", (int)context->local_sequence_number, (int)context->remote_sequence_number); + DEBUG_DUMP_HEX_LABEL("FINISHED OPAQUE", &buf[res], size); + DEBUG_DUMP_HEX_LABEL("VERIFY", out, size); + return TLS_NOT_VERIFIED; + } +#ifdef TLS_ACCEPT_SECURE_RENEGOTIATION + if (size) { + if (context->is_server) { + TLS_FREE(context->verify_data); + context->verify_data = (unsigned char *)TLS_MALLOC(size); + if (context->verify_data) { + memcpy(context->verify_data, out, size); + context->verify_len = size; + } + } else { + // concatenate client verify and server verify + context->verify_data = (unsigned char *)TLS_REALLOC(context->verify_data, size); + if (context->verify_data) { + memcpy(context->verify_data + context->verify_len, out, size); + context->verify_len += size; + } else + context->verify_len = 0; + } + } +#endif + TLS_FREE(out); + } + if (context->is_server) + *write_packets = 3; + else + context->connection_status = 0xFF; + res += size; + return res; +} + +#ifdef WITH_TLS_13 +int tls_parse_verify_tls13(struct TLSContext *context, const unsigned char *buf, int buf_len) { + CHECK_SIZE(7, buf_len, TLS_NEED_MORE_DATA) + unsigned int size = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + + if (size < 2) + return buf_len; + + unsigned char signing_data[TLS_MAX_HASH_SIZE + 98]; + int signing_data_len; + + // first 64 bytes to 0x20 (32) + memset(signing_data, 0x20, 64); + // context string 33 bytes + if (context->is_server) + memcpy(signing_data + 64, "TLS 1.3, client CertificateVerify", 33); + else + memcpy(signing_data + 64, "TLS 1.3, server CertificateVerify", 33); + // a single 0 byte separator + signing_data[97] = 0; + signing_data_len = 98; + + signing_data_len += _private_tls_get_hash(context, signing_data + 98); + DEBUG_DUMP_HEX_LABEL("signature data", signing_data, signing_data_len); + unsigned short signature = ntohs(*(unsigned short *)&buf[3]); + unsigned short signature_size = ntohs(*(unsigned short *)&buf[5]); + int valid = 0; + CHECK_SIZE(7 + signature_size, buf_len, TLS_NEED_MORE_DATA) + switch (signature) { +#ifdef TLS_ECDSA_SUPPORTED + case 0x0403: + // secp256r1 + sha256 + valid = _private_tls_verify_ecdsa(context, sha256, buf + 7, signature_size, signing_data, signing_data_len, &secp256r1); + break; + case 0x0503: + // secp384r1 + sha384 + valid = _private_tls_verify_ecdsa(context, sha384, buf + 7, signature_size, signing_data, signing_data_len, &secp384r1); + break; + case 0x0603: + // secp521r1 + sha512 + valid = _private_tls_verify_ecdsa(context, sha512, buf + 7, signature_size, signing_data, signing_data_len, &secp521r1); + break; +#endif + case 0x0804: + valid = _private_tls_verify_rsa(context, sha256, buf + 7, signature_size, signing_data, signing_data_len); + break; + default: + DEBUG_PRINT("Unsupported signature: %x\n", (int)signature); + return TLS_UNSUPPORTED_CERTIFICATE; + } + if (valid != 1) { + DEBUG_PRINT("Signature FAILED!\n"); + return TLS_DECRYPTION_FAILED; + } + return buf_len; +} +#endif + +int tls_parse_verify(struct TLSContext *context, const unsigned char *buf, int buf_len) { +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + return tls_parse_verify_tls13(context, buf, buf_len); +#endif + CHECK_SIZE(7, buf_len, TLS_BAD_CERTIFICATE) + unsigned int bytes_to_follow = buf[0] * 0x10000 + buf[1] * 0x100 + buf[2]; + CHECK_SIZE(bytes_to_follow, buf_len - 3, TLS_BAD_CERTIFICATE) + int res = -1; + + if ((context->version == TLS_V12) || (context->version == DTLS_V12) || (context->version == TLS_V13) || (context->version == DTLS_V13)) { + unsigned int hash = buf[3]; + unsigned int algorithm = buf[4]; +#ifdef TLS_ECDSA_SUPPORTED + if ((algorithm != rsa) && (algorithm != ecdsa)) { + if (context->dtls == 4) { + DEBUG_PRINT("DTLS-SRTP mode, skipping signature check for unsupported signature (%x/%x)\n", algorithm, hash); + context->client_verified = 1; + return 1; + } + return TLS_UNSUPPORTED_CERTIFICATE; + } +#else + if (algorithm != rsa) + return TLS_UNSUPPORTED_CERTIFICATE; +#endif + unsigned short size = ntohs(*(unsigned short *)&buf[5]); + CHECK_SIZE(size, bytes_to_follow - 4, TLS_BAD_CERTIFICATE) + DEBUG_PRINT("ALGORITHM %i/%i (%i)\n", hash, algorithm, (int)size); + DEBUG_DUMP_HEX_LABEL("VERIFY", &buf[7], bytes_to_follow - 7); + + if (algorithm == rsa) + res = _private_tls_verify_rsa(context, hash, &buf[7], size, context->cached_handshake, context->cached_handshake_len); + else + res = _private_tls_verify_ecdsa(context, hash, &buf[7], size, context->cached_handshake, context->cached_handshake_len, NULL); + } else { +#ifdef TLS_LEGACY_SUPPORT + unsigned short size = ntohs(*(unsigned short *)&buf[3]); + CHECK_SIZE(size, bytes_to_follow - 2, TLS_BAD_CERTIFICATE) + res = _private_tls_verify_rsa(context, md5, &buf[5], size, context->cached_handshake, context->cached_handshake_len); +#endif + } + if (context->cached_handshake) { + // not needed anymore + TLS_FREE(context->cached_handshake); + context->cached_handshake = NULL; + context->cached_handshake_len = 0; + } + if (res == 1) { + DEBUG_PRINT("Signature OK\n"); + context->client_verified = 1; + } else { + DEBUG_PRINT("Signature FAILED\n"); + context->client_verified = 0; + } + return 1; +} + +void _private_dtls_reset_handshake(struct TLSContext *context) { + if ((context) && (context->dtls)) { + // reset state + memset(context->hs_messages, 0, sizeof(context->hs_messages)); + context->connection_status = 0; + context->cipher_spec_set = 0; + context->dtls_seq = 0; +#ifdef TLS_LEGACY_SUPPORT + if (context->cached_handshake) { + TLS_FREE(context->cached_handshake); + context->cached_handshake = NULL; + context->cached_handshake_len = 0; + } +#endif + _private_tls_done_hash(context, NULL); + + if (context->is_server) { + if (context->client_certificates) { + int i; + for (i = 0; i < context->client_certificates_count; i++) + tls_destroy_certificate(context->client_certificates[i]); + TLS_FREE(context->client_certificates); + + context->client_certificates_count = 0; + context->client_certificates = NULL; + } + } + + while (context->dtls_data->dtls_handshake_list) { + struct TLSHandshakeList *next = (struct TLSHandshakeList *)context->dtls_data->dtls_handshake_list->next; + if (context->dtls_data->dtls_handshake_list->msg) { + TLS_FREE(context->dtls_data->dtls_handshake_list->msg); + } + TLS_FREE(context->dtls_data->dtls_handshake_list); + context->dtls_data->dtls_handshake_list = next; + } + context->dtls_data->dtls_handshake_list = NULL; + } +} + +void _private_dtls_rehash(struct TLSContext *context, unsigned char msg_type) { + if ((context) && (context->dtls) && (msg_type)) { + // create a new list, delete old one + struct TLSHandshakeList *handshake_list = context->dtls_data->dtls_handshake_list; + + int found = 0; + while (handshake_list) { + struct TLSHandshakeList *next = (struct TLSHandshakeList *)handshake_list->next; + + if ((handshake_list->direction == 0) && (handshake_list->msg[0] == msg_type)) { + found = 1; + context->connection_status = handshake_list->connection_status; + break; + } + + handshake_list = next; + } + + if (!found) { + DEBUG_PRINT("message already cleared\n"); + return; + } + + handshake_list = context->dtls_data->dtls_handshake_list; + + struct TLSHandshakeList *to_delete = handshake_list; + context->dtls_data->dtls_handshake_list = NULL; + + _private_tls_done_hash(context, NULL); + + while (handshake_list) { + struct TLSHandshakeList *next = (struct TLSHandshakeList *)handshake_list->next; + + + if (handshake_list->direction) + context->connection_status = handshake_list->connection_status; + + if ((handshake_list->direction == 0) && (handshake_list->msg[0] == msg_type)) + break; + + _private_tls_update_hash(context, handshake_list->msg, handshake_list->len, handshake_list->direction, handshake_list->connection_status); + handshake_list = next; + } + + handshake_list = to_delete; + while (handshake_list) { + struct TLSHandshakeList *next = (struct TLSHandshakeList *)handshake_list->next; + if (handshake_list->msg) { + TLS_FREE(handshake_list->msg); + } + TLS_FREE(handshake_list); + handshake_list = next; + } + } +} + +int tls_parse_payload(struct TLSContext *context, const unsigned char *buf, int buf_len, tls_validation_function certificate_verify) { + int orig_len = buf_len; + if (context->connection_status == 0xFF) { +#ifndef TLS_ACCEPT_SECURE_RENEGOTIATION + // renegotiation disabled (emit warning alert) + if (context->dtls) + return orig_len; + _private_tls_write_packet(tls_build_alert(context, 0, no_renegotiation)); + return 1; +#endif + } + + unsigned char local_buffer[DTLS_MAX_FRAGMENT_SIZE + 12]; + while ((buf_len >= 4) && (!context->critical_error)) { + int payload_res = 0; + unsigned char update_hash = 1; + CHECK_SIZE(1, buf_len, TLS_NEED_MORE_DATA) + unsigned char type = buf[0]; + unsigned int write_packets = 0; + unsigned int dtls_cookie_verified = 0; + int certificate_verify_alert = no_error; + unsigned int payload_size = buf[1] * 0x10000 + buf[2] * 0x100 + buf[3] + 3; + if (context->dtls) { + payload_size += 8; + if (context->dtls_data->fragment) { + CHECK_SIZE(payload_size - 11, context->dtls_data->fragment->written, TLS_NEED_MORE_DATA) + + local_buffer[0] = type; + + TLS_24_BIT(local_buffer, 1, context->dtls_data->fragment->written); + + local_buffer[4] = buf[4]; + local_buffer[5] = buf[5]; + + local_buffer[6] = 0; + local_buffer[7] = 0; + local_buffer[8] = 0; + + TLS_24_BIT(local_buffer, 9, context->dtls_data->fragment->written); + + memcpy(local_buffer + 12, context->dtls_data->fragment->buffer, context->dtls_data->fragment->written); + + buf = local_buffer; + buf_len = context->dtls_data->fragment->written + 12; + + TLS_FREE(context->dtls_data->fragment->buffer); + TLS_FREE(context->dtls_data->fragment); + context->dtls_data->fragment = 0; + } else { + CHECK_SIZE(payload_size + 1, buf_len, TLS_NEED_MORE_DATA) + } + } else { + CHECK_SIZE(payload_size + 1, buf_len, TLS_NEED_MORE_DATA) + } + switch (type) { + // hello request + case 0x00: + DEBUG_PRINT(" => HELLO REQUEST (RENEGOTIATION?)\n"); + CHECK_HANDSHAKE_STATE(context, 0, 1); + if (context->dtls) + context->dtls_seq = 0; + if (context->is_server) + payload_res = TLS_UNEXPECTED_MESSAGE; + else { + if (context->connection_status == 0xFF) { + // renegotiation +#ifdef TLS_ACCEPT_SECURE_RENEGOTIATION + if (context->critical_error) + payload_res = TLS_UNEXPECTED_MESSAGE; + else { + _private_tls_reset_context(context); + _private_tls_write_packet(tls_build_hello(context, 0)); + return 1; + } +#else + payload_res = TLS_NO_RENEGOTIATION; +#endif + } else + payload_res = TLS_UNEXPECTED_MESSAGE; + } + // no payload + break; + // client hello + case 0x01: + DEBUG_PRINT(" => CLIENT HELLO\n"); + if (context->dtls) { + _private_dtls_reset_handshake(context); + } else { + CHECK_HANDSHAKE_STATE(context, 1, (context->dtls ? 2 : 1)); + } + if ((context->dtls == 4) && (!context->is_server) && (context->connection_status == 0)) { + DEBUG_PRINT("SRTP HANDSHAKE: SWITCHING FROM CLIENT TO SERVER\n"); + context->is_server = 1; + context->certificates = context->client_certificates; + context->certificates_count = context->client_certificates_count; + context->request_client_certificate = 1; + + context->client_certificates = NULL; + context->client_certificates_count = 0; + } + + if (context->is_server) { + payload_res = tls_parse_hello(context, buf + 1, payload_size, &write_packets, &dtls_cookie_verified); + DEBUG_PRINT(" => DTLS COOKIE VERIFIED: %i (%i)\n", dtls_cookie_verified, payload_res); + if ((context->dtls) && (payload_res > 0) && (!dtls_cookie_verified)) { + // wait client hello + context->connection_status = 3; + update_hash = 0; + } + } else + payload_res = TLS_UNEXPECTED_MESSAGE; + break; + // server hello + case 0x02: + DEBUG_PRINT(" => SERVER HELLO\n"); + CHECK_HANDSHAKE_STATE(context, 2, 1); + if (context->is_server) + payload_res = TLS_UNEXPECTED_MESSAGE; + else + payload_res = tls_parse_hello(context, buf + 1, payload_size, &write_packets, &dtls_cookie_verified); + break; + // hello verify request + case 0x03: + DEBUG_PRINT(" => VERIFY REQUEST\n"); + CHECK_HANDSHAKE_STATE(context, 3, 1); + if ((context->dtls) && (!context->is_server)) { + payload_res = tls_parse_verify_request(context, buf + 1, payload_size, &write_packets); + update_hash = 0; + } else + payload_res = TLS_UNEXPECTED_MESSAGE; + break; + // certificate + case 0x0B: + DEBUG_PRINT(" => CERTIFICATE\n"); + CHECK_HANDSHAKE_STATE(context, 4, 1); +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + if (context->connection_status == 2) { + payload_res = tls_parse_certificate(context, buf + 1, payload_size, context->is_server); + if (context->is_server) { + if ((certificate_verify) && (context->client_certificates_count)) + certificate_verify_alert = certificate_verify(context, context->client_certificates, context->client_certificates_count); + // empty certificates are permitted for client + if (payload_res == 0) + payload_res = 1; + } else { + if ((certificate_verify) && (context->certificates_count)) + certificate_verify_alert = certificate_verify(context, context->certificates, context->certificates_count); + } + } else + payload_res = TLS_UNEXPECTED_MESSAGE; + } else +#endif + if (context->connection_status == 1) { + if (context->is_server) { + // client certificate + payload_res = tls_parse_certificate(context, buf + 1, payload_size, 1); + if ((certificate_verify) && (context->client_certificates_count)) + certificate_verify_alert = certificate_verify(context, context->client_certificates, context->client_certificates_count); + // empty certificates are permitted for client + if (payload_res == 0) + payload_res = 1; + } else { + payload_res = tls_parse_certificate(context, buf + 1, payload_size, 0); + if ((certificate_verify) && (context->certificates_count)) + certificate_verify_alert = certificate_verify(context, context->certificates, context->certificates_count); + } + } else { + DEBUG_PRINT("* UNEXPECTED CERTIFICATE (%i)\n", context->connection_status); + payload_res = TLS_UNEXPECTED_MESSAGE; + } + break; + // server key exchange + case 0x0C: + DEBUG_PRINT(" => SERVER KEY EXCHANGE\n"); + CHECK_HANDSHAKE_STATE(context, 5, 1); + if (context->is_server) + payload_res = TLS_UNEXPECTED_MESSAGE; + else + payload_res = tls_parse_server_key_exchange(context, buf + 1, payload_size); + break; + // certificate request + case 0x0D: + DEBUG_PRINT(" => CERTIFICATE REQUEST\n"); + CHECK_HANDSHAKE_STATE(context, 6, 1); + // server to client + if (context->is_server) + payload_res = TLS_UNEXPECTED_MESSAGE; + else + context->client_verified = 2; + break; + // server hello done + case 0x0E: + DEBUG_PRINT(" => SERVER HELLO DONE\n"); + CHECK_HANDSHAKE_STATE(context, 7, 1); + if (context->is_server) { + payload_res = TLS_UNEXPECTED_MESSAGE; + } else { + payload_res = tls_parse_server_hello_done(context, buf + 1, payload_size); + if (payload_res > 0) + write_packets = 1; + } + break; + // certificate verify + case 0x0F: + DEBUG_PRINT(" => CERTIFICATE VERIFY (%i)\n", payload_size); + CHECK_HANDSHAKE_STATE(context, 8, 1); + if (context->connection_status == 2) + payload_res = tls_parse_verify(context, buf + 1, payload_size); + else + payload_res = TLS_UNEXPECTED_MESSAGE; + break; + // client key exchange + case 0x10: + DEBUG_PRINT(" => CLIENT KEY EXCHANGE\n"); + CHECK_HANDSHAKE_STATE(context, 9, 1); + if (context->is_server) { + _private_tls_update_hash(context, buf, payload_size + 1, 0, 0); + payload_res = tls_parse_client_key_exchange(context, buf + 1, payload_size); + update_hash = 0; + } else + payload_res = TLS_UNEXPECTED_MESSAGE; + break; + // finished + case 0x14: + DEBUG_PRINT(" => FINISHED\n"); + if (context->cached_handshake) { + TLS_FREE(context->cached_handshake); + context->cached_handshake = NULL; + context->cached_handshake_len = 0; + } + CHECK_HANDSHAKE_STATE(context, 10, 1); + payload_res = tls_parse_finished(context, buf + 1, payload_size, &write_packets); + if (payload_res > 0) + memset(context->hs_messages, 0, sizeof(context->hs_messages)); + #ifdef WITH_TLS_13 + if ((!context->is_server) && ((context->version == TLS_V13) || (context->version == DTLS_V13))) { + update_hash = 0; + DEBUG_PRINT("<= SENDING FINISHED\n"); + _private_tls_update_hash(context, buf, payload_size + 1, 0, 0xFF); + _private_tls_write_packet(tls_build_finished(context)); + _private_tls13_key(context, 0); + context->connection_status = 0xFF; + context->local_sequence_number = 0; + context->remote_sequence_number = 0; + } +#endif + break; +#ifdef WITH_TLS_13 + case 0x08: + // encrypted extensions ... ignore it for now + break; +#endif + default: + DEBUG_PRINT(" => NOT UNDERSTOOD PAYLOAD TYPE: %x\n", (int)type); + return TLS_NOT_UNDERSTOOD; + } + if ((type != 0x00) && (update_hash) && (payload_res != TLS_UNEXPECTED_MESSAGE)) + _private_tls_update_hash(context, buf, payload_size + 1, 0, 0); + + if (certificate_verify_alert != no_error) { + _private_tls_write_packet(tls_build_alert(context, 1, certificate_verify_alert)); + context->critical_error = 1; + } + + if (payload_res < 0) { + switch (payload_res) { + case TLS_UNEXPECTED_MESSAGE: + if (context->dtls) + return orig_len; + else + _private_tls_write_packet(tls_build_alert(context, 1, unexpected_message)); + break; + case TLS_COMPRESSION_NOT_SUPPORTED: + _private_tls_write_packet(tls_build_alert(context, 1, decompression_failure)); + break; + case TLS_BROKEN_PACKET: + _private_tls_write_packet(tls_build_alert(context, 1, decode_error)); + break; + case TLS_NO_MEMORY: + _private_tls_write_packet(tls_build_alert(context, 1, internal_error)); + break; + case TLS_NOT_VERIFIED: + _private_tls_write_packet(tls_build_alert(context, 1, bad_record_mac)); + break; + case TLS_BAD_CERTIFICATE: + if (context->is_server) { + // bad client certificate, continue + _private_tls_write_packet(tls_build_alert(context, 0, bad_certificate)); + payload_res = 0; + } else + _private_tls_write_packet(tls_build_alert(context, 1, bad_certificate)); + break; + case TLS_UNSUPPORTED_CERTIFICATE: + _private_tls_write_packet(tls_build_alert(context, 1, unsupported_certificate)); + break; + case TLS_NO_COMMON_CIPHER: + _private_tls_write_packet(tls_build_alert(context, 1, insufficient_security)); + break; + case TLS_NOT_UNDERSTOOD: + _private_tls_write_packet(tls_build_alert(context, 1, internal_error)); + break; + case TLS_NO_RENEGOTIATION: + _private_tls_write_packet(tls_build_alert(context, 0, no_renegotiation)); + payload_res = 0; + break; + case TLS_DECRYPTION_FAILED: + _private_tls_write_packet(tls_build_alert(context, 1, decryption_failed_RESERVED)); + break; + } + if (payload_res < 0) + return payload_res; + } + if (certificate_verify_alert != no_error) + payload_res = TLS_BAD_CERTIFICATE; + + // except renegotiation + switch (write_packets) { + case 1: + if (context->client_verified == 2) { + DEBUG_PRINT("<= Building CERTIFICATE \n"); + _private_tls_write_packet(tls_build_certificate(context)); + context->client_verified = 0; + } + // client handshake + DEBUG_PRINT("<= Building KEY EXCHANGE\n"); + _private_tls_write_packet(tls_build_client_key_exchange(context)); + DEBUG_PRINT("<= Building CHANGE CIPHER SPEC\n"); + _private_tls_write_packet(tls_build_change_cipher_spec(context)); + context->cipher_spec_set = 1; + context->local_sequence_number = 0; + DEBUG_PRINT("<= Building CLIENT FINISHED\n"); + _private_tls_write_packet(tls_build_finished(context)); + context->cipher_spec_set = 0; +#ifdef TLS_12_FALSE_START + if ((!context->is_server) && (context->version == TLS_V12)) { + // https://tools.ietf.org/html/rfc7918 + // 5.1. Symmetric Cipher + // Clients MUST NOT use the False Start protocol modification in a + // handshake unless the cipher suite uses a symmetric cipher that is + // considered cryptographically strong. + switch (context->cipher) { + case TLS_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: + case TLS_DHE_RSA_WITH_AES_256_GCM_SHA384: + case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: + context->false_start = 1; + break; + } + } +#endif + break; + case 2: + // server handshake + if ((context->dtls) && (dtls_cookie_verified == 0)) { + _private_tls_write_packet(tls_build_verify_request(context)); + _private_dtls_reset(context); + } else { + DEBUG_PRINT("<= SENDING SERVER HELLO\n"); +#ifdef WITH_TLS_13 + if (context->connection_status == 3) { + context->connection_status = 2; + _private_tls_write_packet(tls_build_hello(context, 0)); + _private_tls_write_packet(tls_build_change_cipher_spec(context)); + _private_tls13_key(context, 1); + context->cipher_spec_set = 1; + DEBUG_PRINT("<= SENDING ENCRYPTED EXTENSIONS\n"); + _private_tls_write_packet(tls_build_encrypted_extensions(context)); + if (context->request_client_certificate) { + DEBUG_PRINT("<= SENDING CERTIFICATE REQUEST\n"); + _private_tls_write_packet(tls_certificate_request(context)); + } + DEBUG_PRINT("<= SENDING CERTIFICATE\n"); + _private_tls_write_packet(tls_build_certificate(context)); + DEBUG_PRINT("<= SENDING CERTIFICATE VERIFY\n"); + _private_tls_write_packet(tls_build_certificate_verify(context)); + DEBUG_PRINT("<= SENDING FINISHED\n"); + _private_tls_write_packet(tls_build_finished(context)); + // new key + TLS_FREE(context->server_finished_hash); + context->server_finished_hash = (unsigned char *)TLS_MALLOC(_private_tls_mac_length(context)); + if (context->server_finished_hash) + _private_tls_get_hash(context, context->server_finished_hash); + break; + } +#endif + _private_tls_write_packet(tls_build_hello(context, 0)); + DEBUG_PRINT("<= SENDING CERTIFICATE\n"); + _private_tls_write_packet(tls_build_certificate(context)); + int ephemeral_cipher = tls_cipher_is_ephemeral(context); + if (ephemeral_cipher) { + DEBUG_PRINT("<= SENDING EPHEMERAL DH KEY\n"); + _private_tls_write_packet(tls_build_server_key_exchange(context, ephemeral_cipher == 1 ? KEA_dhe_rsa : KEA_ec_diffie_hellman)); + } + if (context->request_client_certificate) { + DEBUG_PRINT("<= SENDING CERTIFICATE REQUEST\n"); + _private_tls_write_packet(tls_certificate_request(context)); + } + DEBUG_PRINT("<= SENDING DONE\n"); + _private_tls_write_packet(tls_build_done(context)); + } + break; + case 3: + // finished + _private_tls_write_packet(tls_build_change_cipher_spec(context)); + _private_tls_write_packet(tls_build_finished(context)); + context->connection_status = 0xFF; + break; + case 4: + // dtls only + context->dtls_seq = 1; + _private_tls_write_packet(tls_build_hello(context, 0)); + break; +#ifdef WITH_TLS_13 + case 5: + // hello retry request + DEBUG_PRINT("<= SENDING HELLO RETRY REQUEST\n"); + _private_tls_write_packet(tls_build_hello(context, 0)); + break; +#endif + } + payload_size ++; + buf += payload_size; + buf_len -= payload_size; + } + return orig_len; +} + +unsigned int _private_tls_hmac_message(unsigned char local, struct TLSContext *context, const unsigned char *buf, int buf_len, const unsigned char *buf2, int buf_len2, unsigned char *out, unsigned int outlen, uint64_t remote_sequence_number) { + hmac_state hash; + int mac_size = outlen; + int hash_idx; + if (mac_size == TLS_SHA1_MAC_SIZE) + hash_idx = find_hash("sha1"); + else + if (mac_size == TLS_SHA384_MAC_SIZE) + hash_idx = find_hash("sha384"); + else + hash_idx = find_hash("sha256"); + + if (hmac_init(&hash, hash_idx, local ? context->crypto.ctx_local_mac.local_mac : context->crypto.ctx_remote_mac.remote_mac, mac_size)) + return 0; + + uint64_t squence_number; + if (context->dtls) + squence_number = htonll(remote_sequence_number); + else + if (local) + squence_number = htonll(context->local_sequence_number); + else + squence_number = htonll(context->remote_sequence_number); + + if (hmac_process(&hash, (unsigned char *)&squence_number, sizeof(uint64_t))) + return 0; + + if (hmac_process(&hash, buf, buf_len)) + return 0; + if ((buf2) && (buf_len2)) { + if (hmac_process(&hash, buf2, buf_len2)) + return 0; + } + unsigned long ref_outlen = outlen; + if (hmac_done(&hash, out, &ref_outlen)) + return 0; + + return (unsigned int)ref_outlen; +} + +int tls_parse_message(struct TLSContext *context, unsigned char *buf, int buf_len, tls_validation_function certificate_verify) { + int res = 5; + if (context->dtls) + res = 13; + int header_size = res; + int payload_res = 0; + + CHECK_SIZE(res, buf_len, TLS_NEED_MORE_DATA) + + unsigned char type = *buf; + + int buf_pos = 1; + + unsigned short version = ntohs(*(unsigned short *)&buf[buf_pos]); + buf_pos += 2; + + uint64_t dtls_sequence_number = 0; + unsigned short dtls_epoch = 0; + if (context->dtls) { + CHECK_SIZE(buf_pos + 8, buf_len, TLS_NEED_MORE_DATA) + dtls_epoch = ntohs(*(unsigned short *)&buf[buf_pos]); + dtls_sequence_number = ntohll(*(uint64_t *)&buf[buf_pos]); + + buf_pos += 8; + } + + VERSION_SUPPORTED(version, TLS_NOT_SAFE) + unsigned short length; + length = ntohs(*(unsigned short *)&buf[buf_pos]); + buf_pos += 2; + + unsigned char *pt = NULL; + const unsigned char *ptr = buf + buf_pos; + + CHECK_SIZE(buf_pos + length, buf_len, TLS_NEED_MORE_DATA) + DEBUG_PRINT("Message type: %0x, length: %i\n", (int)type, (int)length); + if ((context->dtls) && (type == TLS_HANDSHAKE)) { + if (!dtls_epoch) + context->cipher_spec_set = 0; + + DEBUG_PRINT("HANDSHAKE RETRANSMISSION DETECTED\n"); + } + if ((context->cipher_spec_set) && (type != TLS_CHANGE_CIPHER)) { + DEBUG_DUMP_HEX_LABEL("encrypted", &buf[header_size], length); + if (!context->crypto.created) { + DEBUG_PRINT("Encryption context not created\n"); + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + return TLS_BROKEN_PACKET; + } + pt = (unsigned char *)TLS_MALLOC(length); + if (!pt) { + DEBUG_PRINT("Error in TLS_MALLOC (%i bytes)\n", (int)length); + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + return TLS_NO_MEMORY; + } + + unsigned char aad[16]; + int aad_size = sizeof(aad); + unsigned char *sequence = aad; + + if (context->crypto.created == 2) { + int delta = 8; + int pt_length; + unsigned char iv[TLS_13_AES_GCM_IV_LENGTH]; + gcm_reset(&context->crypto.ctx_remote.aes_gcm_remote); + +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + aad[0] = TLS_APPLICATION_DATA; + aad[1] = 0x03; + aad[2] = 0x03; + *((unsigned short *)(aad + 3)) = htons(buf_len - header_size); + aad_size = 5; + sequence = aad + 5; + if (context->dtls) + *((uint64_t *)sequence) = *(uint64_t *)(buf + 3); + else + *((uint64_t *)sequence) = htonll(context->remote_sequence_number); + memcpy(iv, context->crypto.ctx_remote_mac.remote_iv, TLS_13_AES_GCM_IV_LENGTH); + int i; + int offset = TLS_13_AES_GCM_IV_LENGTH - 8; + for (i = 0; i < 8; i++) + iv[offset + i] = context->crypto.ctx_remote_mac.remote_iv[offset + i] ^ sequence[i]; + pt_length = buf_len - header_size - TLS_GCM_TAG_LEN; + delta = 0; + } else { +#endif + aad_size = 13; + pt_length = length - 8 - TLS_GCM_TAG_LEN; + // build aad and iv + if (context->dtls) + *((uint64_t *)aad) = htonll(dtls_sequence_number); + else + *((uint64_t *)aad) = htonll(context->remote_sequence_number); + aad[8] = buf[0]; + aad[9] = buf[1]; + aad[10] = buf[2]; + + memcpy(iv, context->crypto.ctx_remote_mac.remote_aead_iv, 4); + memcpy(iv + 4, buf + header_size, 8); + *((unsigned short *)(aad + 11)) = htons((unsigned short)pt_length); +#ifdef WITH_TLS_13 + } +#endif + if (pt_length < 0) { + DEBUG_PRINT("Invalid packet length"); + TLS_FREE(pt); + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + return TLS_BROKEN_PACKET; + } + DEBUG_DUMP_HEX_LABEL("aad", aad, aad_size); + DEBUG_DUMP_HEX_LABEL("aad iv", iv, 12); + + int res0 = gcm_add_iv(&context->crypto.ctx_remote.aes_gcm_remote, iv, 12); + int res1 = gcm_add_aad(&context->crypto.ctx_remote.aes_gcm_remote, aad, aad_size); + memset(pt, 0, length); + DEBUG_PRINT("PT SIZE: %i\n", pt_length); + int res2 = gcm_process(&context->crypto.ctx_remote.aes_gcm_remote, pt, pt_length, buf + header_size + delta, GCM_DECRYPT); + unsigned char tag[32]; + unsigned long taglen = 32; + int res3 = gcm_done(&context->crypto.ctx_remote.aes_gcm_remote, tag, &taglen); + if ((res0) || (res1) || (res2) || (res3) || (taglen != TLS_GCM_TAG_LEN)) { + DEBUG_PRINT("ERROR: gcm_add_iv: %i, gcm_add_aad: %i, gcm_process: %i, gcm_done: %i\n", res0, res1, res2, res3); + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + return TLS_BROKEN_PACKET; + } + DEBUG_DUMP_HEX_LABEL("decrypted", pt, pt_length); + DEBUG_DUMP_HEX_LABEL("tag", tag, taglen); + // check tag + if (memcmp(buf + header_size + delta + pt_length, tag, taglen)) { + DEBUG_PRINT("INTEGRITY CHECK FAILED (msg length %i)\n", pt_length); + DEBUG_DUMP_HEX_LABEL("TAG RECEIVED", buf + header_size + delta + pt_length, taglen); + DEBUG_DUMP_HEX_LABEL("TAG COMPUTED", tag, taglen); + TLS_FREE(pt); + + // silently ignore packet for DTLS + if (context->dtls) + return header_size + length; + + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + _private_tls_write_packet(tls_build_alert(context, 1, bad_record_mac)); + return TLS_INTEGRITY_FAILED; + } + ptr = pt; + length = (unsigned short)pt_length; +#ifdef TLS_WITH_CHACHA20_POLY1305 + } else + if (context->crypto.created == 3) { + int pt_length = length - POLY1305_TAGLEN; + unsigned int counter = 1; + unsigned char poly1305_key[POLY1305_KEYLEN]; + unsigned char trail[16]; + unsigned char mac_tag[POLY1305_TAGLEN]; + aad_size = 16; + if (pt_length < 0) { + DEBUG_PRINT("Invalid packet length"); + TLS_FREE(pt); + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + return TLS_BROKEN_PACKET; + } +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + aad[0] = TLS_APPLICATION_DATA; + aad[1] = 0x03; + aad[2] = 0x03; + *((unsigned short *)(aad + 3)) = htons(buf_len - header_size); + aad_size = 5; + sequence = aad + 5; + if (context->dtls) + *((uint64_t *)sequence) = *(uint64_t *)(buf + 3); + else + *((uint64_t *)sequence) = htonll(context->remote_sequence_number); + } else { +#endif + if (context->dtls) + *((uint64_t *)aad) = htonll(dtls_sequence_number); + else + *((uint64_t *)aad) = htonll(context->remote_sequence_number); + aad[8] = buf[0]; + aad[9] = buf[1]; + aad[10] = buf[2]; + *((unsigned short *)(aad + 11)) = htons((unsigned short)pt_length); + aad[13] = 0; + aad[14] = 0; + aad[15] = 0; +#ifdef WITH_TLS_13 + } +#endif + + chacha_ivupdate(&context->crypto.ctx_remote.chacha_remote, context->crypto.ctx_remote_mac.remote_aead_iv, sequence, (unsigned char *)&counter); + + chacha_encrypt_bytes(&context->crypto.ctx_remote.chacha_remote, buf + header_size, pt, pt_length); + DEBUG_DUMP_HEX_LABEL("decrypted", pt, pt_length); + ptr = pt; + length = (unsigned short)pt_length; + + chacha20_poly1305_key(&context->crypto.ctx_remote.chacha_remote, poly1305_key); + poly1305_context ctx; + _private_tls_poly1305_init(&ctx, poly1305_key); + _private_tls_poly1305_update(&ctx, aad, aad_size); + static unsigned char zeropad[] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; + int rem = aad_size % 16; + if (rem) + _private_tls_poly1305_update(&ctx, zeropad, 16 - rem); + _private_tls_poly1305_update(&ctx, buf + header_size, pt_length); + rem = pt_length % 16; + if (rem) + _private_tls_poly1305_update(&ctx, zeropad, 16 - rem); + + _private_tls_U32TO8(&trail[0], aad_size == 5 ? 5 : 13); + *(int *)&trail[4] = 0; + _private_tls_U32TO8(&trail[8], pt_length); + *(int *)&trail[12] = 0; + + _private_tls_poly1305_update(&ctx, trail, 16); + _private_tls_poly1305_finish(&ctx, mac_tag); + if (memcmp(mac_tag, buf + header_size + pt_length, POLY1305_TAGLEN)) { + DEBUG_PRINT("INTEGRITY CHECK FAILED (msg length %i)\n", length); + DEBUG_DUMP_HEX_LABEL("POLY1305 TAG RECEIVED", buf + header_size + pt_length, POLY1305_TAGLEN); + DEBUG_DUMP_HEX_LABEL("POLY1305 TAG COMPUTED", mac_tag, POLY1305_TAGLEN); + TLS_FREE(pt); + + // silently ignore packet for DTLS + if (context->dtls) + return header_size + length; + + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + _private_tls_write_packet(tls_build_alert(context, 1, bad_record_mac)); + return TLS_INTEGRITY_FAILED; + } +#endif + } else { + int err = _private_tls_crypto_decrypt(context, buf + header_size, pt, length); + if (err) { + TLS_FREE(pt); + DEBUG_PRINT("Decryption error %i\n", (int)err); + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + return TLS_BROKEN_PACKET; + } + unsigned char padding_byte = pt[length - 1]; + unsigned char padding = padding_byte + 1; + + // poodle check + int padding_index = length - padding; + if (padding_index > 0) { + int i; + int limit = length - 1; + for (i = length - padding; i < limit; i++) { + if (pt[i] != padding_byte) { + TLS_FREE(pt); + DEBUG_PRINT("BROKEN PACKET (POODLE ?)\n"); + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + _private_tls_write_packet(tls_build_alert(context, 1, decrypt_error)); + return TLS_BROKEN_PACKET; + } + } + } + + unsigned int decrypted_length = length; + if (padding < decrypted_length) + decrypted_length -= padding; + + DEBUG_DUMP_HEX_LABEL("decrypted", pt, decrypted_length); + ptr = pt; +#ifdef TLS_LEGACY_SUPPORT + if ((context->version != TLS_V10) && (decrypted_length > TLS_AES_IV_LENGTH)) { + decrypted_length -= TLS_AES_IV_LENGTH; + ptr += TLS_AES_IV_LENGTH; + } +#else + if (decrypted_length > TLS_AES_IV_LENGTH) { + decrypted_length -= TLS_AES_IV_LENGTH; + ptr += TLS_AES_IV_LENGTH; + } +#endif + length = decrypted_length; + + unsigned int mac_size = _private_tls_mac_length(context); + if ((length < mac_size) || (!mac_size)) { + TLS_FREE(pt); + DEBUG_PRINT("BROKEN PACKET\n"); + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + _private_tls_write_packet(tls_build_alert(context, 1, decrypt_error)); + return TLS_BROKEN_PACKET; + } + + length -= mac_size; + + const unsigned char *message_hmac = &ptr[length]; + unsigned char hmac_out[TLS_MAX_MAC_SIZE]; + unsigned char temp_buf[5]; + memcpy(temp_buf, buf, 3); + *(unsigned short *)(temp_buf + 3) = htons(length); + unsigned int hmac_out_len = _private_tls_hmac_message(0, context, temp_buf, 5, ptr, length, hmac_out, mac_size, dtls_sequence_number); + if ((hmac_out_len != mac_size) || (memcmp(message_hmac, hmac_out, mac_size))) { + DEBUG_PRINT("INTEGRITY CHECK FAILED (msg length %i)\n", length); + DEBUG_DUMP_HEX_LABEL("HMAC RECEIVED", message_hmac, mac_size); + DEBUG_DUMP_HEX_LABEL("HMAC COMPUTED", hmac_out, hmac_out_len); + TLS_FREE(pt); + + // silently ignore packet for DTLS + if (context->dtls) + return header_size + length; + + _private_random_sleep(context, TLS_MAX_ERROR_SLEEP_uS); + _private_tls_write_packet(tls_build_alert(context, 1, bad_record_mac)); + + return TLS_INTEGRITY_FAILED; + } + } + } + if (context->dtls) { + context->dtls_epoch_remote = dtls_epoch; + context->remote_sequence_number = dtls_sequence_number & 0xFFFFFFFFFFFFLL; + } else + context->remote_sequence_number ++; + +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + if (/*(context->connection_status == 2) && */(type == TLS_APPLICATION_DATA) && (context->crypto.created)) { + do { + length--; + type = ptr[length]; + } while (!type); + } + } +#endif + switch (type) { + // application data + case TLS_APPLICATION_DATA: + if (context->connection_status != 0xFF) { + DEBUG_PRINT("UNEXPECTED APPLICATION DATA MESSAGE\n"); + payload_res = TLS_UNEXPECTED_MESSAGE; + _private_tls_write_packet(tls_build_alert(context, 1, unexpected_message)); + } else { + DEBUG_PRINT("APPLICATION DATA MESSAGE (TLS VERSION: %x):\n", (int)context->version); + DEBUG_DUMP(ptr, length); + DEBUG_PRINT("\n"); + _private_tls_write_app_data(context, ptr, length); + } + break; + // handshake + case TLS_HANDSHAKE: + DEBUG_PRINT("HANDSHAKE MESSAGE\n"); + payload_res = tls_parse_payload(context, ptr, length, certificate_verify); + break; + // change cipher spec + case TLS_CHANGE_CIPHER: + context->dtls_epoch_remote ++; + if ((context->connection_status != 2) && (!context->dtls)) { +#ifdef WITH_TLS_13 + if (context->connection_status == 4) { + DEBUG_PRINT("IGNORING CHANGE CIPHER SPEC MESSAGE (HELLO RETRY REQUEST)\n"); + break; + } +#endif + DEBUG_PRINT("UNEXPECTED CHANGE CIPHER SPEC MESSAGE (%i)\n", context->connection_status); + _private_tls_write_packet(tls_build_alert(context, 1, unexpected_message)); + payload_res = TLS_UNEXPECTED_MESSAGE; + } else { + DEBUG_PRINT("CHANGE CIPHER SPEC MESSAGE\n"); + context->cipher_spec_set = 1; + // reset sequence numbers + context->remote_sequence_number = 0; + } +#ifdef WITH_TLS_13 + if (!context->is_server) + _private_tls13_key(context, 1); +#endif + break; + // alert + case TLS_ALERT: + DEBUG_PRINT("ALERT MESSAGE\n"); + if (length >= 2) { + DEBUG_DUMP_HEX(ptr, length); + int level = ptr[0]; + int code = ptr[1]; + if (level == TLS_ALERT_CRITICAL) { + context->critical_error = 1; + res = TLS_ERROR_ALERT; + } + context->error_code = code; + } + break; + default: + DEBUG_PRINT("NOT UNDERSTOOD MESSAGE TYPE: %x\n", (int)type); + TLS_FREE(pt); + return TLS_NOT_UNDERSTOOD; + } + TLS_FREE(pt); + + if (payload_res < 0) + return payload_res; + + if (res > 0) + return header_size + length; + + return res; +} + +unsigned int asn1_get_len(const unsigned char *buffer, int buf_len, unsigned int *octets) { + *octets = 0; + + if (buf_len < 1) + return 0; + + unsigned char size = buffer[0]; + int i; + if (size & 0x80) { + *octets = size & 0x7F; + if ((int)*octets > buf_len - 1) + return 0; + // max 32 bits + unsigned int ref_octets = *octets; + if (*octets > 4) + ref_octets = 4; + if ((int)*octets > buf_len -1) + return 0; + unsigned int long_size = 0; + unsigned int coef = 1; + + for (i = ref_octets; i > 0; i--) { + long_size += buffer[i] * coef; + coef *= 0x100; + } + ++*octets; + return long_size; + } + ++*octets; + return size; +} + +void print_index(const unsigned int *fields) { + int i = 0; + while (fields[i]) { + if (i) + DEBUG_PRINT("."); + DEBUG_PRINT("%i", fields[i]); + i++; + } + while (i < 6) { + DEBUG_PRINT(" "); + i++; + } +} + +int _is_field(const unsigned int *fields, const unsigned int *prefix) { + int i = 0; + while (prefix[i]) { + if (fields[i] != prefix[i]) + return 0; + i++; + } + return 1; +} + +int _private_tls_hash_len(int algorithm) { + switch (algorithm) { + case TLS_RSA_SIGN_MD5: + return 16; + case TLS_RSA_SIGN_SHA1: + return 20; + case TLS_RSA_SIGN_SHA256: + case TLS_ECDSA_SIGN_SHA256: + return 32; + case TLS_RSA_SIGN_SHA384: + return 48; + case TLS_RSA_SIGN_SHA512: + return 64; + } + return 0; +} + +unsigned char *_private_tls_compute_hash(int algorithm, const unsigned char *message, unsigned int message_len) { + unsigned char *hash = NULL; + if ((!message) || (!message_len)) + return hash; + int err; + hash_state state; + switch (algorithm) { + case TLS_RSA_SIGN_MD5: + DEBUG_PRINT("SIGN MD5\n"); + hash = (unsigned char *)TLS_MALLOC(16); + if (!hash) + return NULL; + + err = md5_init(&state); + if (!err) { + err = md5_process(&state, message, message_len); + if (!err) + err = md5_done(&state, hash); + } + break; + case TLS_RSA_SIGN_SHA1: + DEBUG_PRINT("SIGN SHA1\n"); + hash = (unsigned char *)TLS_MALLOC(20); + if (!hash) + return NULL; + + err = sha1_init(&state); + if (!err) { + err = sha1_process(&state, message, message_len); + if (!err) + err = sha1_done(&state, hash); + } + break; + case TLS_RSA_SIGN_SHA256: + case TLS_ECDSA_SIGN_SHA256: + DEBUG_PRINT("SIGN SHA256\n"); + hash = (unsigned char *)TLS_MALLOC(32); + if (!hash) + return NULL; + + err = sha256_init(&state); + if (!err) { + err = sha256_process(&state, message, message_len); + if (!err) + err = sha256_done(&state, hash); + } + break; + case TLS_RSA_SIGN_SHA384: + DEBUG_PRINT("SIGN SHA384\n"); + hash = (unsigned char *)TLS_MALLOC(48); + if (!hash) + return NULL; + + err = sha384_init(&state); + if (!err) { + err = sha384_process(&state, message, message_len); + if (!err) + err = sha384_done(&state, hash); + } + break; + case TLS_RSA_SIGN_SHA512: + DEBUG_PRINT("SIGN SHA512\n"); + hash = (unsigned char *)TLS_MALLOC(64); + if (!hash) + return NULL; + + err = sha512_init(&state); + if (!err) { + err = sha512_process(&state, message, message_len); + if (!err) + err = sha512_done(&state, hash); + } + break; + default: + DEBUG_PRINT("UNKNOWN SIGNATURE ALGORITHM\n"); + } + return hash; +} + +int tls_certificate_verify_signature(struct TLSCertificate *cert, struct TLSCertificate *parent) { + if ((!cert) || (!parent) || (!cert->sign_key) || (!cert->fingerprint) || (!cert->sign_len) || (!parent->der_bytes) || (!parent->der_len)) { + DEBUG_PRINT("CANNOT VERIFY SIGNATURE"); + return 0; + } + tls_init(); + int hash_len = _private_tls_hash_len(cert->algorithm); + if (hash_len <= 0) + return 0; + + int hash_index = -1; + switch (cert->algorithm) { + case TLS_RSA_SIGN_MD5: + hash_index = find_hash("md5"); + break; + case TLS_RSA_SIGN_SHA1: + hash_index = find_hash("sha1"); + break; + case TLS_RSA_SIGN_SHA256: + case TLS_ECDSA_SIGN_SHA256: + hash_index = find_hash("sha256"); + break; + case TLS_RSA_SIGN_SHA384: + hash_index = find_hash("sha384"); + break; + case TLS_RSA_SIGN_SHA512: + hash_index = find_hash("sha512"); + break; + default: + DEBUG_PRINT("UNKNOWN SIGNATURE ALGORITHM\n"); + return 0; + } +#ifdef TLS_ECDSA_SUPPORTED + if (cert->algorithm == TLS_ECDSA_SIGN_SHA256) { + ecc_key key; + int err = ecc_import(parent->der_bytes, parent->der_len, &key); + if (err) { + DEBUG_PRINT("Error importing ECC certificate (code: %i)\n", err); + DEBUG_DUMP_HEX_LABEL("CERTIFICATE", parent->der_bytes, parent->der_len); + return 0; + } + int ecc_stat = 0; + unsigned char *signature = cert->sign_key; + int signature_len = cert->sign_len; + if (!signature[0]) { + signature++; + signature_len--; + } + err = ecc_verify_hash(signature, signature_len, cert->fingerprint, hash_len, &ecc_stat, &key); + ecc_free(&key); + if (err) { + DEBUG_PRINT("ECC HASH VERIFY ERROR %i\n", err); + return 0; + } + DEBUG_PRINT("ECC CERTIFICATE VALIDATION: %i\n", ecc_stat); + return ecc_stat; + } +#endif + + rsa_key key; + int err = rsa_import(parent->der_bytes, parent->der_len, &key); + if (err) { + DEBUG_PRINT("Error importing RSA certificate (code: %i)\n", err); + DEBUG_DUMP_HEX_LABEL("CERTIFICATE", parent->der_bytes, parent->der_len); + return 0; + } + int rsa_stat = 0; + unsigned char *signature = cert->sign_key; + int signature_len = cert->sign_len; + if (!signature[0]) { + signature++; + signature_len--; + } + err = rsa_verify_hash_ex(signature, signature_len, cert->fingerprint, hash_len, LTC_PKCS_1_V1_5, hash_index, hash_len, &rsa_stat, &key); + rsa_free(&key); + if (err) { + DEBUG_PRINT("HASH VERIFY ERROR %i\n", err); + return 0; + } + DEBUG_PRINT("CERTIFICATE VALIDATION: %i\n", rsa_stat); + return rsa_stat; +} + +int tls_certificate_chain_is_valid(struct TLSCertificate **certificates, int len) { + if ((!certificates) || (!len)) + return bad_certificate; + + int i; + len--; + + // expired certificate or not yet valid ? + if (tls_certificate_is_valid(certificates[0])) + return bad_certificate; + + // check + for (i = 0; i < len; i++) { + // certificate in chain is expired ? + if (tls_certificate_is_valid(certificates[i+1])) + return bad_certificate; + if (!tls_certificate_verify_signature(certificates[i], certificates[i+1])) + return bad_certificate; + } + return 0; +} + +int tls_certificate_chain_is_valid_root(struct TLSContext *context, struct TLSCertificate **certificates, int len) { + if ((!certificates) || (!len) || (!context->root_certificates) || (!context->root_count)) + return bad_certificate; + int i; + unsigned int j; + for (i = 0; i < len; i++) { + for (j = 0; j < context->root_count; j++) { + // check if root certificate expired + if (tls_certificate_is_valid(context->root_certificates[j])) + continue; + // if any root validates any certificate in the chain, then is root validated + if (tls_certificate_verify_signature(certificates[i], context->root_certificates[j])) + return 0; + } + } + return bad_certificate; +} + +int _private_is_oid(struct _private_OID_chain *ref_chain, const unsigned char *looked_oid, int looked_oid_len) { + while (ref_chain) { + if (ref_chain->oid) { + if (_is_oid2(ref_chain->oid, looked_oid, 16, looked_oid_len)) + return 1; + } + ref_chain = (struct _private_OID_chain *)ref_chain->top; + } + return 0; +} + +int _private_asn1_parse(struct TLSContext *context, struct TLSCertificate *cert, const unsigned char *buffer, unsigned int size, int level, unsigned int *fields, unsigned char *has_key, int client_cert, unsigned char *top_oid, struct _private_OID_chain *chain) { + struct _private_OID_chain local_chain; + local_chain.top = chain; + unsigned int pos = 0; + // X.690 + int idx = 0; + unsigned char oid[16]; + memset(oid, 0, 16); + local_chain.oid = oid; + if (has_key) + *has_key = 0; + unsigned char local_has_key = 0; + const unsigned char *cert_data = NULL; + unsigned int cert_len = 0; + while (pos < size) { + unsigned int start_pos = pos; + CHECK_SIZE(2, size - pos, TLS_NEED_MORE_DATA) + unsigned char first = buffer[pos++]; + unsigned char type = first & 0x1F; + unsigned char constructed = first & 0x20; + unsigned char element_class = first >> 6; + unsigned int octets = 0; + unsigned int temp; + idx++; + if (level <= TLS_ASN1_MAXLEVEL) + fields[level - 1] = idx; + unsigned int length = asn1_get_len((unsigned char *)&buffer[pos], size - pos, &octets); + if ((octets > 4) || (octets > size - pos)) { + DEBUG_PRINT("CANNOT READ CERTIFICATE\n"); + return pos; + } + pos += octets; + CHECK_SIZE(length, size - pos, TLS_NEED_MORE_DATA) + //DEBUG_PRINT("FIRST: %x => %x (%i)\n", (int)first, (int)type, length); + // sequence + //DEBUG_PRINT("%2i: ", level); +#ifdef DEBUG + DEBUG_INDEX(fields); + int i1; + for (i1 = 1; i1 < level; i1++) + DEBUG_PRINT(" "); +#endif + + if ((length) && (constructed)) { + switch (type) { + case 0x03: + DEBUG_PRINT("CONSTRUCTED BITSTREAM\n"); + break; + case 0x10: + DEBUG_PRINT("SEQUENCE\n"); + if ((level == 2) && (idx == 1)) { + cert_len = length + (pos - start_pos); + cert_data = &buffer[start_pos]; + } + // private key on server or public key on client + if ((!cert->version) && (_is_field(fields, priv_der_id))) { + TLS_FREE(cert->der_bytes); + temp = length + (pos - start_pos); + cert->der_bytes = (unsigned char *)TLS_MALLOC(temp); + if (cert->der_bytes) { + memcpy(cert->der_bytes, &buffer[start_pos], temp); + cert->der_len = temp; + } else + cert->der_len = 0; + } + break; + case 0x11: + DEBUG_PRINT("EMBEDDED PDV\n"); + break; + case 0x00: + if (element_class == 0x02) { + DEBUG_PRINT("CONTEXT-SPECIFIC\n"); + break; + } + default: + DEBUG_PRINT("CONSTRUCT TYPE %02X\n", (int)type); + } + local_has_key = 0; + _private_asn1_parse(context, cert, &buffer[pos], length, level + 1, fields, &local_has_key, client_cert, top_oid, &local_chain); + if ((((local_has_key) && (context) && ((!context->is_server) || (client_cert))) || (!context)) && (_is_field(fields, pk_id))) { + TLS_FREE(cert->der_bytes); + temp = length + (pos - start_pos); + cert->der_bytes = (unsigned char *)TLS_MALLOC(temp); + if (cert->der_bytes) { + memcpy(cert->der_bytes, &buffer[start_pos], temp); + cert->der_len = temp; + } else + cert->der_len = 0; + } + } else { + switch (type) { + case 0x00: + // end of content + DEBUG_PRINT("END OF CONTENT\n"); + return pos; + break; + case 0x01: + // boolean + temp = buffer[pos]; + DEBUG_PRINT("BOOLEAN: %i\n", temp); + break; + case 0x02: + // integer + if (_is_field(fields, pk_id)) { + if (has_key) + *has_key = 1; + + if (idx == 1) + tls_certificate_set_key(cert, &buffer[pos], length); + else + if (idx == 2) + tls_certificate_set_exponent(cert, &buffer[pos], length); + } else + if (_is_field(fields, serial_id)) + tls_certificate_set_serial(cert, &buffer[pos], length); + if (_is_field(fields, version_id)) { + if (length == 1) + cert->version = buffer[pos]; +#ifdef TLS_X509_V1_SUPPORT + else + cert->version = 0; + idx++; +#endif + } + if (level >= 2) { + unsigned int fields_temp[3]; + fields_temp[0] = fields[level - 2]; + fields_temp[1] = fields[level - 1]; + fields_temp[2] = 0; + if (_is_field(fields_temp, priv_id)) + tls_certificate_set_priv(cert, &buffer[pos], length); + } + DEBUG_PRINT("INTEGER(%i): ", length); + DEBUG_DUMP_HEX(&buffer[pos], length); + if ((chain) && (length > 2)) { + if (_private_is_oid(chain, san_oid, sizeof(san_oid) - 1)) { + cert->san = (unsigned char **)TLS_REALLOC(cert->san, sizeof(unsigned char *) * (cert->san_length + 1)); + if (cert->san) { + cert->san[cert->san_length] = NULL; + tls_certificate_set_copy(&cert->san[cert->san_length], &buffer[pos], length); + DEBUG_PRINT(" => SUBJECT ALTERNATIVE NAME: %s", cert->san[cert->san_length ]); + cert->san_length++; + } else + cert->san_length = 0; + } + } + DEBUG_PRINT("\n"); + break; + case 0x03: + if (_is_field(fields, pk_id)) { + if (has_key) + *has_key = 1; + } + // bitstream + DEBUG_PRINT("BITSTREAM(%i): ", length); + DEBUG_DUMP_HEX(&buffer[pos], length); + DEBUG_PRINT("\n"); + if (_is_field(fields, sign_id)) { + tls_certificate_set_sign_key(cert, &buffer[pos], length); + } else + if ((cert->ec_algorithm) && (_is_field(fields, pk_id))) { + tls_certificate_set_key(cert, &buffer[pos], length); + } else { + if ((buffer[pos] == 0x00) && (length > 256)) + _private_asn1_parse(context, cert, &buffer[pos]+1, length - 1, level + 1, fields, &local_has_key, client_cert, top_oid, &local_chain); + else + _private_asn1_parse(context, cert, &buffer[pos], length, level + 1, fields, &local_has_key, client_cert, top_oid, &local_chain); +#ifdef TLS_FORWARD_SECRECY + #ifdef TLS_ECDSA_SUPPORTED + if (top_oid) { + if (_is_oid2(top_oid, TLS_EC_prime256v1_OID, sizeof(oid), sizeof(TLS_EC_prime256v1) - 1)) { + cert->ec_algorithm = secp256r1.iana; + } else + if (_is_oid2(top_oid, TLS_EC_secp224r1_OID, sizeof(oid), sizeof(TLS_EC_secp224r1_OID) - 1)) { + cert->ec_algorithm = secp224r1.iana; + } else + if (_is_oid2(top_oid, TLS_EC_secp384r1_OID, sizeof(oid), sizeof(TLS_EC_secp384r1_OID) - 1)) { + cert->ec_algorithm = secp384r1.iana; + } else + if (_is_oid2(top_oid, TLS_EC_secp521r1_OID, sizeof(oid), sizeof(TLS_EC_secp521r1_OID) - 1)) { + cert->ec_algorithm = secp521r1.iana; + } + if ((cert->ec_algorithm) && (!cert->pk)) + tls_certificate_set_key(cert, &buffer[pos], length); + } + #endif +#endif + } + break; + case 0x04: + if ((top_oid) && (_is_field(fields, ecc_priv_id)) && (!cert->priv)) { + DEBUG_PRINT("BINARY STRING(%i): ", length); + DEBUG_DUMP_HEX(&buffer[pos], length); + DEBUG_PRINT("\n"); + tls_certificate_set_priv(cert, &buffer[pos], length); + } else + _private_asn1_parse(context, cert, &buffer[pos], length, level + 1, fields, &local_has_key, client_cert, top_oid, &local_chain); + break; + case 0x05: + DEBUG_PRINT("NULL\n"); + break; + case 0x06: + // object identifier + if (_is_field(fields, pk_id)) { +#ifdef TLS_ECDSA_SUPPORTED + if ((length == 8) || (length == 5)) + tls_certificate_set_algorithm(context, &cert->ec_algorithm, &buffer[pos], length); + else +#endif + tls_certificate_set_algorithm(context, &cert->key_algorithm, &buffer[pos], length); + } + if (_is_field(fields, algorithm_id)) + tls_certificate_set_algorithm(context, &cert->algorithm, &buffer[pos], length); + + DEBUG_PRINT("OBJECT IDENTIFIER(%i): ", length); + DEBUG_DUMP_HEX(&buffer[pos], length); + DEBUG_PRINT("\n"); + // check previous oid + if (_is_oid2(oid, ocsp_oid, 16, sizeof(ocsp_oid) - 1)) + tls_certificate_set_copy(&cert->ocsp, &buffer[pos], length); + + if (length < 16) + memcpy(oid, &buffer[pos], length); + else + memcpy(oid, &buffer[pos], 16); + if (top_oid) + memcpy(top_oid, oid, 16); + break; + case 0x09: + DEBUG_PRINT("REAL NUMBER(%i): ", length); + DEBUG_DUMP_HEX(&buffer[pos], length); + DEBUG_PRINT("\n"); + break; + case 0x17: + // utc time + DEBUG_PRINT("UTC TIME: ["); + DEBUG_DUMP(&buffer[pos], length); + DEBUG_PRINT("]\n"); + + if (_is_field(fields, validity_id)) { + if (idx == 1) + tls_certificate_set_copy_date(&cert->not_before, &buffer[pos], length); + else + tls_certificate_set_copy_date(&cert->not_after, &buffer[pos], length); + } + break; + case 0x18: + // generalized time + DEBUG_PRINT("GENERALIZED TIME: ["); + DEBUG_DUMP(&buffer[pos], length); + DEBUG_PRINT("]\n"); + break; + case 0x13: + // printable string + case 0x0C: + case 0x14: + case 0x15: + case 0x16: + case 0x19: + case 0x1A: + case 0x1B: + case 0x1C: + case 0x1D: + case 0x1E: + if (_is_field(fields, issurer_id)) { + if (_is_oid(oid, country_oid, 3)) + tls_certificate_set_copy(&cert->issuer_country, &buffer[pos], length); + else + if (_is_oid(oid, state_oid, 3)) + tls_certificate_set_copy(&cert->issuer_state, &buffer[pos], length); + else + if (_is_oid(oid, location_oid, 3)) + tls_certificate_set_copy(&cert->issuer_location, &buffer[pos], length); + else + if (_is_oid(oid, entity_oid, 3)) + tls_certificate_set_copy(&cert->issuer_entity, &buffer[pos], length); + else + if (_is_oid(oid, subject_oid, 3)) + tls_certificate_set_copy(&cert->issuer_subject, &buffer[pos], length); + } else + if (_is_field(fields, owner_id)) { + if (_is_oid(oid, country_oid, 3)) + tls_certificate_set_copy(&cert->country, &buffer[pos], length); + else + if (_is_oid(oid, state_oid, 3)) + tls_certificate_set_copy(&cert->state, &buffer[pos], length); + else + if (_is_oid(oid, location_oid, 3)) + tls_certificate_set_copy(&cert->location, &buffer[pos], length); + else + if (_is_oid(oid, entity_oid, 3)) + tls_certificate_set_copy(&cert->entity, &buffer[pos], length); + else + if (_is_oid(oid, subject_oid, 3)) + tls_certificate_set_copy(&cert->subject, &buffer[pos], length); + } + DEBUG_PRINT("STR: ["); + DEBUG_DUMP(&buffer[pos], length); + DEBUG_PRINT("]\n"); + break; + case 0x10: + DEBUG_PRINT("EMPTY SEQUENCE\n"); + break; + case 0xA: + DEBUG_PRINT("ENUMERATED(%i): ", length); + DEBUG_DUMP_HEX(&buffer[pos], length); + DEBUG_PRINT("\n"); + break; + default: + DEBUG_PRINT("========> NOT SUPPORTED %x\n", (int)type); + // not supported / needed + break; + } + } + pos += length; + } + if ((level == 2) && (cert->sign_key) && (cert->sign_len) && (cert_len) && (cert_data)) { + TLS_FREE(cert->fingerprint); + cert->fingerprint = _private_tls_compute_hash(cert->algorithm, cert_data, cert_len); +#ifdef DEBUG + if (cert->fingerprint) { + DEBUG_DUMP_HEX_LABEL("FINGERPRINT", cert->fingerprint, _private_tls_hash_len(cert->algorithm)); + } +#endif + } + return pos; +} + +struct TLSCertificate *asn1_parse(struct TLSContext *context, const unsigned char *buffer, unsigned int size, int client_cert) { + unsigned int fields[TLS_ASN1_MAXLEVEL]; + memset(fields, 0, sizeof(int) * TLS_ASN1_MAXLEVEL); + struct TLSCertificate *cert = tls_create_certificate(); + if (cert) { + if (client_cert < 0) { + client_cert = 0; + // private key + unsigned char top_oid[16]; + memset(top_oid, 0, sizeof(top_oid)); + _private_asn1_parse(context, cert, buffer, size, 1, fields, NULL, client_cert, top_oid, NULL); + } else + _private_asn1_parse(context, cert, buffer, size, 1, fields, NULL, client_cert, NULL, NULL); + } + return cert; +} + +int tls_load_certificates(struct TLSContext *context, const unsigned char *pem_buffer, int pem_size) { + if (!context) + return TLS_GENERIC_ERROR; + + unsigned int len; + int idx = 0; + do { + unsigned char *data = tls_pem_decode(pem_buffer, pem_size, idx++, &len); + if ((!data) || (!len)) + break; + struct TLSCertificate *cert = asn1_parse(context, data, len, 0); + if (cert) { + if ((cert->version == 2) +#ifdef TLS_X509_V1_SUPPORT + || (cert->version == 0) +#endif + ) { + TLS_FREE(cert->der_bytes); + cert->der_bytes = data; + cert->der_len = len; + data = NULL; + if (cert->priv) { + DEBUG_PRINT("WARNING - parse error (private key encountered in certificate)\n"); + TLS_FREE(cert->priv); + cert->priv = NULL; + cert->priv_len = 0; + } + if (context->is_server) { + context->certificates = (struct TLSCertificate **)TLS_REALLOC(context->certificates, (context->certificates_count + 1) * sizeof(struct TLSCertificate *)); + context->certificates[context->certificates_count] = cert; + context->certificates_count++; + DEBUG_PRINT("Loaded certificate: %i\n", (int)context->certificates_count); + } else { + context->client_certificates = (struct TLSCertificate **)TLS_REALLOC(context->client_certificates, (context->client_certificates_count + 1) * sizeof(struct TLSCertificate *)); + context->client_certificates[context->client_certificates_count] = cert; + context->client_certificates_count++; + DEBUG_PRINT("Loaded client certificate: %i\n", (int)context->client_certificates_count); + } + } else { + DEBUG_PRINT("WARNING - certificate version error (v%i)\n", (int)cert->version); + tls_destroy_certificate(cert); + } + } + TLS_FREE(data); + } while (1); + + if (context->is_server) + return context->certificates_count; + + return context->client_certificates_count; +} + +int tls_load_private_key(struct TLSContext *context, const unsigned char *pem_buffer, int pem_size) { + if (!context) + return TLS_GENERIC_ERROR; + + unsigned int len; + int idx = 0; + do { + unsigned char *data = tls_pem_decode(pem_buffer, pem_size, idx++, &len); + if ((!data) || (!len)) + break; + struct TLSCertificate *cert = asn1_parse(context, data, len, -1); + if (cert) { + if (!cert->der_len) { + TLS_FREE(cert->der_bytes); + cert->der_bytes = data; + cert->der_len = len; + } else + TLS_FREE(data); + if ((cert) && (cert->priv) && (cert->priv_len)) { +#ifdef TLS_ECDSA_SUPPORTED + if (cert->ec_algorithm) { + DEBUG_PRINT("Loaded ECC private key\n"); + if (context->ec_private_key) + tls_destroy_certificate(context->ec_private_key); + context->ec_private_key = cert; + return 1; + } else +#endif + { + DEBUG_PRINT("Loaded private key\n"); + if (context->private_key) + tls_destroy_certificate(context->private_key); + context->private_key = cert; + return 1; + } + } + tls_destroy_certificate(cert); + } else + TLS_FREE(data); + } while (1); + return 0; +} + +int tls_clear_certificates(struct TLSContext *context) { + unsigned int i; + if ((!context) || (!context->is_server) || (context->is_child)) + return TLS_GENERIC_ERROR; + + if (context->root_certificates) { + for (i = 0; i < context->root_count; i++) + tls_destroy_certificate(context->root_certificates[i]); + } + context->root_certificates = NULL; + context->root_count = 0; + if (context->private_key) + tls_destroy_certificate(context->private_key); + context->private_key = NULL; +#ifdef TLS_ECDSA_SUPPORTED + if (context->ec_private_key) + tls_destroy_certificate(context->ec_private_key); + context->ec_private_key = NULL; +#endif + TLS_FREE(context->certificates); + context->certificates = NULL; + context->certificates_count = 0; + return 0; +} + +#ifdef WITH_TLS_13 +struct TLSPacket *tls_build_certificate_verify(struct TLSContext *context) { + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, context->version, 0); + //certificate verify + tls_packet_uint8(packet, 0x0F); + unsigned int size_offset = packet->len; + tls_packet_uint24(packet, 0); + + unsigned char out[TLS_MAX_RSA_KEY]; +#ifdef TLS_ECDSA_SUPPORTED + unsigned long out_len = TLS_MAX_RSA_KEY; +#endif + + unsigned char signing_data[TLS_MAX_HASH_SIZE + 98]; + int signing_data_len; + + // first 64 bytes to 0x20 (32) + memset(signing_data, 0x20, 64); + // context string 33 bytes + if (context->is_server) + memcpy(signing_data + 64, "TLS 1.3, server CertificateVerify", 33); + else + memcpy(signing_data + 64, "TLS 1.3, client CertificateVerify", 33); + // a single 0 byte separator + signing_data[97] = 0; + signing_data_len = 98; + + signing_data_len += _private_tls_get_hash(context, signing_data + 98); + DEBUG_DUMP_HEX_LABEL("verify data", signing_data, signing_data_len); + int hash_algorithm = sha256; +#ifdef TLS_ECDSA_SUPPORTED + if (tls_is_ecdsa(context)) { + switch (context->ec_private_key->ec_algorithm) { + case 23: + // secp256r1 + sha256 + tls_packet_uint16(packet, 0x0403); + break; + case 24: + // secp384r1 + sha384 + tls_packet_uint16(packet, 0x0503); + hash_algorithm = sha384; + break; + case 25: + // secp521r1 + sha512 + tls_packet_uint16(packet, 0x0603); + hash_algorithm = sha512; + break; + default: + DEBUG_PRINT("UNSUPPORTED CURVE (SIGNING)\n"); + packet->broken = 1; + return packet; + } + } else +#endif + { + tls_packet_uint16(packet, 0x0804); + } + + int packet_size = 2; +#ifdef TLS_ECDSA_SUPPORTED + if (tls_is_ecdsa(context)) { + if (_private_tls_sign_ecdsa(context, hash_algorithm, signing_data, signing_data_len, out, &out_len) == 1) { + DEBUG_PRINT("ECDSA signing OK! (ECDSA, length %lu)\n", out_len); + tls_packet_uint16(packet, out_len); + tls_packet_append(packet, out, out_len); + packet_size += out_len + 2; + } + } else +#endif + if (_private_tls_sign_rsa(context, hash_algorithm, signing_data, signing_data_len, out, &out_len) == 1) { + DEBUG_PRINT("RSA signing OK! (length %lu)\n", out_len); + tls_packet_uint16(packet, out_len); + tls_packet_append(packet, out, out_len); + packet_size += out_len + 2; + } + packet->buf[size_offset] = packet_size / 0x10000; + packet_size %= 0x10000; + packet->buf[size_offset + 1] = packet_size / 0x100; + packet_size %= 0x100; + packet->buf[size_offset + 2] = packet_size; + + tls_packet_update(packet); + return packet; +} +#endif + +struct TLSPacket *tls_build_certificate(struct TLSContext *context) { + int i; + unsigned int all_certificate_size = 0; + int certificates_count; + struct TLSCertificate **certificates; + if (context->is_server) { + certificates_count = context->certificates_count; + certificates = context->certificates; + } else { + certificates_count = context->client_certificates_count; + certificates = context->client_certificates; + } + int delta = 3; +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + delta = 5; +#endif +#ifdef TLS_ECDSA_SUPPORTED + int is_ecdsa = tls_is_ecdsa(context); + if (is_ecdsa) { + for (i = 0; i < certificates_count; i++) { + struct TLSCertificate *cert = certificates[i]; + if ((cert) && (cert->der_len) && (cert->ec_algorithm)) + all_certificate_size += cert->der_len + delta; + } + if (!all_certificate_size) { + for (i = 0; i < certificates_count; i++) { + struct TLSCertificate *cert = certificates[i]; + if ((cert) && (cert->der_len)) + all_certificate_size += cert->der_len + delta; + } + } + } else { + for (i = 0; i < certificates_count; i++) { + struct TLSCertificate *cert = certificates[i]; + if ((cert) && (cert->der_len) && (!cert->ec_algorithm)) + all_certificate_size += cert->der_len + delta; + } + } +#else + for (i = 0; i < certificates_count; i++) { + struct TLSCertificate *cert = certificates[i]; + if ((cert) && (cert->der_len)) + all_certificate_size += cert->der_len + delta; + } +#endif + if (!all_certificate_size) { + DEBUG_PRINT("NO CERTIFICATE SET\n"); + } + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, context->version, 0); + tls_packet_uint8(packet, 0x0B); + if (all_certificate_size) { +#ifdef WITH_TLS_13 + // context + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + tls_packet_uint24(packet, all_certificate_size + 4); + tls_packet_uint8(packet, 0); + } else +#endif + tls_packet_uint24(packet, all_certificate_size + 3); + + if (context->dtls) + _private_dtls_handshake_data(context, packet, all_certificate_size + 3); + + tls_packet_uint24(packet, all_certificate_size); + for (i = 0; i < certificates_count; i++) { + struct TLSCertificate *cert = certificates[i]; + if ((cert) && (cert->der_len)) { +#ifdef TLS_ECDSA_SUPPORTED + // is RSA certificate ? + if ((is_ecdsa) && (!cert->ec_algorithm)) + continue; + // is ECC certificate ? + if ((!is_ecdsa) && (cert->ec_algorithm)) + continue; +#endif + // 2 times -> one certificate + tls_packet_uint24(packet, cert->der_len); + tls_packet_append(packet, cert->der_bytes, cert->der_len); +#ifdef WITH_TLS_13 + // extension + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + tls_packet_uint16(packet, 0); +#endif + } + } + } else { + tls_packet_uint24(packet, all_certificate_size); +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + tls_packet_uint8(packet, 0); +#endif + + if (context->dtls) + _private_dtls_handshake_data(context, packet, all_certificate_size); + } + tls_packet_update(packet); + if (context->dtls) + context->dtls_seq++; + return packet; +} + +#ifdef WITH_TLS_13 +struct TLSPacket *tls_build_encrypted_extensions(struct TLSContext *context) { + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, context->version, 3); + tls_packet_uint8(packet, 0x08); + if (context->negotiated_alpn) { + int alpn_negotiated_len = strlen(context->negotiated_alpn); + int alpn_len = alpn_negotiated_len + 1; + + tls_packet_uint24(packet, alpn_len + 8); + tls_packet_uint16(packet, alpn_len + 6); + tls_packet_uint16(packet, 0x10); + tls_packet_uint16(packet, alpn_len + 2); + tls_packet_uint16(packet, alpn_len); + + tls_packet_uint8(packet, alpn_negotiated_len); + tls_packet_append(packet, (unsigned char *)context->negotiated_alpn, alpn_negotiated_len); + } else { + tls_packet_uint24(packet, 2); + tls_packet_uint16(packet, 0); + } + tls_packet_update(packet); + return packet; +} +#endif + +struct TLSPacket *tls_build_finished(struct TLSContext *context) { + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, context->version, TLS_MIN_FINISHED_OPAQUE_LEN + 64); + tls_packet_uint8(packet, 0x14); +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) + tls_packet_uint24(packet, _private_tls_mac_length(context)); + else +#endif + tls_packet_uint24(packet, TLS_MIN_FINISHED_OPAQUE_LEN); + if (context->dtls) + _private_dtls_handshake_data(context, packet, TLS_MIN_FINISHED_OPAQUE_LEN); + // verify + unsigned char hash[TLS_MAX_HASH_SIZE]; + unsigned long out_size = TLS_MIN_FINISHED_OPAQUE_LEN; +#ifdef WITH_TLS_13 + unsigned char out[TLS_MAX_HASH_SIZE]; +#else + unsigned char out[TLS_MIN_FINISHED_OPAQUE_LEN]; +#endif + unsigned int hash_len; + + // server verifies client's message + if (context->is_server) { +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + hash_len = _private_tls_get_hash(context, hash); + if ((!context->finished_key) || (!hash_len)) { + DEBUG_PRINT("NO FINISHED KEY COMPUTED OR NO HANDSHAKE HASH\n"); + packet->broken = 1; + return packet; + } + + DEBUG_DUMP_HEX_LABEL("HS HASH", hash, hash_len); + DEBUG_DUMP_HEX_LABEL("HS FINISH", context->finished_key, hash_len); + DEBUG_DUMP_HEX_LABEL("HS REMOTE FINISH", context->remote_finished_key, hash_len); + + out_size = hash_len; + hmac_state hmac; + hmac_init(&hmac, _private_tls_get_hash_idx(context), context->finished_key, hash_len); + hmac_process(&hmac, hash, hash_len); + hmac_done(&hmac, out, &out_size); + } else +#endif + { + hash_len = _private_tls_done_hash(context, hash); + _private_tls_prf(context, out, TLS_MIN_FINISHED_OPAQUE_LEN, context->master_key, context->master_key_len, (unsigned char *)"server finished", 15, hash, hash_len, NULL, 0); + _private_tls_destroy_hash(context); + } + } else { +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + hash_len = _private_tls_get_hash(context, hash); + if ((!context->finished_key) || (!hash_len)) { + DEBUG_PRINT("NO FINISHED KEY COMPUTED OR NO HANDSHAKE HASH\n"); + packet->broken = 1; + return packet; + } + DEBUG_DUMP_HEX_LABEL("HS HASH", hash, hash_len); + DEBUG_DUMP_HEX_LABEL("HS FINISH", context->finished_key, hash_len); + DEBUG_DUMP_HEX_LABEL("HS REMOTE FINISH", context->remote_finished_key, hash_len); + + TLS_FREE(context->server_finished_hash); + context->server_finished_hash = (unsigned char *)TLS_MALLOC(hash_len); + if (context->server_finished_hash) + memcpy(context->server_finished_hash, hash, hash_len); + + out_size = hash_len; + hmac_state hmac; + hmac_init(&hmac, _private_tls_get_hash_idx(context), context->finished_key, hash_len); + hmac_process(&hmac, hash, hash_len); + hmac_done(&hmac, out, &out_size); + } else { +#endif + hash_len = _private_tls_get_hash(context, hash); + _private_tls_prf(context, out, TLS_MIN_FINISHED_OPAQUE_LEN, context->master_key, context->master_key_len, (unsigned char *)"client finished", 15, hash, hash_len, NULL, 0); +#ifdef WITH_TLS_13 + } +#endif + } + tls_packet_append(packet, out, out_size); + tls_packet_update(packet); + DEBUG_DUMP_HEX_LABEL("VERIFY DATA", out, out_size); +#ifdef TLS_ACCEPT_SECURE_RENEGOTIATION + if (context->is_server) { + // concatenate client verify and server verify + context->verify_data = (unsigned char *)TLS_REALLOC(context->verify_data, out_size); + if (context->verify_data) { + memcpy(context->verify_data + context->verify_len, out, out_size); + context->verify_len += out_size; + } else + context->verify_len = 0; + } else { + TLS_FREE(context->verify_data); + context->verify_data = (unsigned char *)TLS_MALLOC(out_size); + if (context->verify_data) { + memcpy(context->verify_data, out, out_size); + context->verify_len = out_size; + } + } +#endif + return packet; +} + +struct TLSPacket *tls_build_change_cipher_spec(struct TLSContext *context) { + struct TLSPacket *packet = tls_create_packet(context, TLS_CHANGE_CIPHER, context->version, 64); + tls_packet_uint8(packet, 1); + tls_packet_update(packet); + context->local_sequence_number = 0; + return packet; +} + +struct TLSPacket *tls_build_done(struct TLSContext *context) { + struct TLSPacket *packet = tls_create_packet(context, TLS_HANDSHAKE, context->version, 0); + tls_packet_uint8(packet, 0x0E); + tls_packet_uint24(packet, 0); + if (context->dtls) { + _private_dtls_handshake_data(context, packet, 0); + context->dtls_seq++; + } + tls_packet_update(packet); + return packet; +} + +struct TLSPacket *tls_build_message(struct TLSContext *context, const unsigned char *data, unsigned int len) { + if ((!data) || (!len)) + return 0; + struct TLSPacket *packet = tls_create_packet(context, TLS_APPLICATION_DATA, context->version, len); + tls_packet_append(packet, data, len); + tls_packet_update(packet); + return packet; +} + +int tls_client_connect(struct TLSContext *context) { + if ((context->is_server) || (context->critical_error)) + return TLS_UNEXPECTED_MESSAGE; + + return _private_tls_write_packet(tls_build_hello(context, 0)); +} + +int tls_write(struct TLSContext *context, const unsigned char *data, unsigned int len) { + if (!context) + return TLS_GENERIC_ERROR; +#ifdef TLS_12_FALSE_START + if ((context->connection_status != 0xFF) && ((context->is_server) || (context->version != TLS_V12) || (context->critical_error) || (!context->false_start))) + return TLS_UNEXPECTED_MESSAGE; +#else + if (context->connection_status != 0xFF) + return TLS_UNEXPECTED_MESSAGE; +#endif + if (len > TLS_MAXTLS_APP_SIZE) + len = TLS_MAXTLS_APP_SIZE; + int actually_written = _private_tls_write_packet(tls_build_message(context, data, len)); + if (actually_written <= 0) + return actually_written; + return len; +} + +struct TLSPacket *tls_build_alert(struct TLSContext *context, char critical, unsigned char code) { + struct TLSPacket *packet = tls_create_packet(context, TLS_ALERT, context->version, 0); + tls_packet_uint8(packet, critical ? TLS_ALERT_CRITICAL : TLS_ALERT_WARNING); + if (critical) + context->critical_error = 1; + tls_packet_uint8(packet, code); + tls_packet_update(packet); + return packet; +} + +int _private_tls_read_from_file(const char *fname, void *buf, int max_len) { + FILE *f = fopen(fname, "rb"); + if (f) { + int size = (int)fread(buf, 1, max_len, f); + fclose(f); + return size; + } + return 0; +} + +int tls_connection_status(struct TLSContext *context) { + return context->connection_status; +} + +int tls_consume_stream(struct TLSContext *context, const unsigned char *buf, int buf_len, tls_validation_function certificate_verify) { + if (!context) + return TLS_GENERIC_ERROR; + + if (context->critical_error) + return TLS_BROKEN_CONNECTION; + + if (buf_len <= 0) { + DEBUG_PRINT("tls_consume_stream called with buf_len %i\n", buf_len); + return 0; + } + + if (!buf) { + DEBUG_PRINT("tls_consume_stream called NULL buffer\n"); + context->critical_error = 1; + return TLS_NO_MEMORY; + } + + unsigned int orig_len = context->message_buffer_len; + context->message_buffer_len += buf_len; + context->message_buffer = (unsigned char *)TLS_REALLOC(context->message_buffer, context->message_buffer_len); + if (!context->message_buffer) { + context->message_buffer_len = 0; + return TLS_NO_MEMORY; + } + memcpy(context->message_buffer + orig_len, buf, buf_len); + unsigned int index = 0; + unsigned int tls_buffer_len = context->message_buffer_len; + int err_flag = 0; + + int tls_header_size; + int tls_size_offset; + + if (context->dtls) { + tls_size_offset = 11; + tls_header_size = 13; + } else { + tls_size_offset = 3; + tls_header_size = 5; + } + while (tls_buffer_len >= tls_header_size) { + unsigned int length = ntohs(*(unsigned short *)&context->message_buffer[index + tls_size_offset]) + tls_header_size; + if (length > tls_buffer_len) { + DEBUG_PRINT("NEED DATA: %i/%i\n", length, tls_buffer_len); + break; + } + int parse_message = 1; + int consumed = 0; + if ((context->dtls) && (!context->cipher_spec_set)) { + // check fragmented! + unsigned char *buffer = &context->message_buffer[index]; + if ((buffer[0] == TLS_HANDSHAKE) && (length > 13)) { + buffer += tls_header_size; + + unsigned int data_length = buffer[1] * 0x10000 + buffer[2] * 0x100 + buffer[3]; + unsigned int fragment_offset = buffer[6] * 0x10000 + buffer[7] * 0x100 + buffer[8]; + unsigned int fragment_length = buffer[9] * 0x10000 + buffer[10] * 0x100 + buffer[11]; + + if ((data_length > DTLS_MAX_FRAGMENT_SIZE) || (fragment_offset + fragment_length > data_length)) { + DEBUG_PRINT("INVALID PACKET SIZE: %i, FRAGMENT OFFSET: %i, FRAGMENT LENGTH: %i\n"); + return TLS_BROKEN_PACKET; + } + + if (data_length != fragment_length) { + // fragmented! + if (!context->dtls_data->fragment) { + context->dtls_data->fragment = (struct DTLSFragment *)TLS_MALLOC(sizeof(struct DTLSFragment)); + if (context->dtls_data->fragment) + memset(context->dtls_data->fragment, 0, sizeof(struct DTLSFragment)); + } + + if (!context->dtls_data->fragment) + return TLS_NO_MEMORY; + + char *fragment_buffer = context->dtls_data->fragment->buffer; + + fragment_buffer = (char *)TLS_REALLOC(fragment_buffer, data_length * sizeof(char *)); + if (!fragment_buffer) + return TLS_NO_MEMORY; + + memcpy(fragment_buffer + fragment_offset, &buffer[12], fragment_length); + context->dtls_data->fragment->buffer = fragment_buffer; + context->dtls_data->fragment->len = data_length; + context->dtls_data->fragment->written += fragment_length; + + if (context->dtls_data->fragment->written != context->dtls_data->fragment->len) { + consumed = length; + parse_message = 0; + } + } + } + } + if (parse_message) + consumed = tls_parse_message(context, &context->message_buffer[index], length, certificate_verify); + + DEBUG_PRINT("Consumed %i/%i bytes\n", consumed, tls_buffer_len); + if (consumed < 0) { + if (!context->critical_error) + context->critical_error = 1; + err_flag = consumed; + break; + } + index += length; + tls_buffer_len -= length; + if (context->critical_error) { + err_flag = TLS_BROKEN_CONNECTION; + break; + } + } + if (err_flag) { + DEBUG_PRINT("ERROR IN CONSUME: %i\n", err_flag); + context->message_buffer_len = 0; + TLS_FREE(context->message_buffer); + context->message_buffer = NULL; + return err_flag; + } + if (index) { + context->message_buffer_len -= index; + if (context->message_buffer_len) { + // no realloc here + memmove(context->message_buffer, context->message_buffer + index, context->message_buffer_len); + } else { + TLS_FREE(context->message_buffer); + context->message_buffer = NULL; + } + } + return index; +} + +void tls_close_notify(struct TLSContext *context) { + if ((!context) || (context->critical_error)) + return; + context->critical_error = 1; + DEBUG_PRINT("CLOSE\n"); + _private_tls_write_packet(tls_build_alert(context, 0, close_notify)); +} + +void tls_alert(struct TLSContext *context, unsigned char critical, int code) { + if (!context) + return; + if ((!context->critical_error) && (critical)) + context->critical_error = 1; + DEBUG_PRINT("ALERT\n"); + _private_tls_write_packet(tls_build_alert(context, critical, code)); +} + +int tls_pending(struct TLSContext *context) { + if (!context->message_buffer) + return 0; + return context->message_buffer_len; +} + +void tls_make_exportable(struct TLSContext *context, unsigned char exportable_flag) { + context->exportable = exportable_flag; + if (!exportable_flag) { + // zero the memory + if ((context->exportable_keys) && (context->exportable_size)) + memset(context->exportable_keys, 0, context->exportable_size); + // free the memory, if alocated + TLS_FREE(context->exportable_keys); + context->exportable_size = 0; + } +} + +int tls_export_context(struct TLSContext *context, unsigned char *buffer, unsigned int buf_len, unsigned char small_version) { + // only negotiated AND exportable connections may be exported + if ((!context) || (context->critical_error) || (context->connection_status != 0xFF) || (!context->exportable) || (!context->exportable_keys) || (!context->exportable_size) || (!context->crypto.created)) { + DEBUG_PRINT("CANNOT EXPORT CONTEXT %i\n", (int)context->connection_status); + return 0; + } + + struct TLSPacket *packet = tls_create_packet(NULL, TLS_SERIALIZED_OBJECT, context->version, 0); + // export buffer version + tls_packet_uint8(packet, 0x01); + tls_packet_uint8(packet, context->connection_status); + tls_packet_uint16(packet, context->cipher); + if (context->is_child) + tls_packet_uint8(packet, 2); + else + tls_packet_uint8(packet, context->is_server); + + if (context->crypto.created == 2) { + // aead +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + tls_packet_uint8(packet, TLS_13_AES_GCM_IV_LENGTH); + tls_packet_append(packet, context->crypto.ctx_local_mac.local_iv, TLS_13_AES_GCM_IV_LENGTH); + tls_packet_append(packet, context->crypto.ctx_remote_mac.remote_iv, TLS_13_AES_GCM_IV_LENGTH); + } else { +#endif + tls_packet_uint8(packet, TLS_AES_GCM_IV_LENGTH); + tls_packet_append(packet, context->crypto.ctx_local_mac.local_aead_iv, TLS_AES_GCM_IV_LENGTH); + tls_packet_append(packet, context->crypto.ctx_remote_mac.remote_aead_iv, TLS_AES_GCM_IV_LENGTH); +#ifdef WITH_TLS_13 + } +#endif +#ifdef TLS_WITH_CHACHA20_POLY1305 + } else + if (context->crypto.created == 3) { + // ChaCha20 + tls_packet_uint8(packet, TLS_CHACHA20_IV_LENGTH); + tls_packet_append(packet, context->crypto.ctx_local_mac.local_nonce, TLS_CHACHA20_IV_LENGTH); + tls_packet_append(packet, context->crypto.ctx_remote_mac.remote_nonce, TLS_CHACHA20_IV_LENGTH); +#endif + } else { + unsigned char iv[TLS_AES_IV_LENGTH]; + unsigned long len = TLS_AES_IV_LENGTH; + + memset(iv, 0, TLS_AES_IV_LENGTH); + cbc_getiv(iv, &len, &context->crypto.ctx_local.aes_local); + tls_packet_uint8(packet, TLS_AES_IV_LENGTH); + tls_packet_append(packet, iv, len); + + memset(iv, 0, TLS_AES_IV_LENGTH); + cbc_getiv(iv, &len, &context->crypto.ctx_remote.aes_remote); + tls_packet_append(packet, iv, TLS_AES_IV_LENGTH); + } + + tls_packet_uint8(packet, context->exportable_size); + tls_packet_append(packet, context->exportable_keys, context->exportable_size); + + if (context->crypto.created == 2) { + tls_packet_uint8(packet, 0); +#ifdef TLS_WITH_CHACHA20_POLY1305 + } else + if (context->crypto.created == 3) { + // ChaCha20 + tls_packet_uint8(packet, 0); + unsigned int i; + for (i = 0; i < 16; i++) + tls_packet_uint32(packet, context->crypto.ctx_local.chacha_local.input[i]); + for (i = 0; i < 16; i++) + tls_packet_uint32(packet, context->crypto.ctx_remote.chacha_remote.input[i]); + tls_packet_append(packet, context->crypto.ctx_local.chacha_local.ks, CHACHA_BLOCKLEN); + tls_packet_append(packet, context->crypto.ctx_remote.chacha_remote.ks, CHACHA_BLOCKLEN); +#endif + } else { + unsigned char mac_length = (unsigned char)_private_tls_mac_length(context); + tls_packet_uint8(packet, mac_length); + tls_packet_append(packet, context->crypto.ctx_local_mac.local_mac, mac_length); + tls_packet_append(packet, context->crypto.ctx_remote_mac.remote_mac, mac_length); + } + + if (small_version) { + tls_packet_uint16(packet, 0); + } else { + tls_packet_uint16(packet, context->master_key_len); + tls_packet_append(packet, context->master_key, context->master_key_len); + } + + uint64_t sequence_number = htonll(context->local_sequence_number); + tls_packet_append(packet, (unsigned char *)&sequence_number, sizeof(uint64_t)); + sequence_number = htonll(context->remote_sequence_number); + tls_packet_append(packet, (unsigned char *)&sequence_number, sizeof(uint64_t)); + + tls_packet_uint32(packet, context->tls_buffer_len); + tls_packet_append(packet, context->tls_buffer, context->tls_buffer_len); + + tls_packet_uint32(packet, context->message_buffer_len); + tls_packet_append(packet, context->message_buffer, context->message_buffer_len); + + tls_packet_uint32(packet, context->application_buffer_len); + tls_packet_append(packet, context->application_buffer, context->application_buffer_len); + tls_packet_uint8(packet, context->dtls); + if (context->dtls) { + tls_packet_uint16(packet, context->dtls_epoch_local); + tls_packet_uint16(packet, context->dtls_epoch_remote); + } + tls_packet_update(packet); + unsigned int size = packet->len; + if ((buffer) && (buf_len)) { + if (size > buf_len) { + tls_destroy_packet(packet); + DEBUG_PRINT("EXPORT BUFFER TO SMALL\n"); + return (int)buf_len - (int)size; + } + memcpy(buffer, packet->buf, size); + } + tls_destroy_packet(packet); + return size; +} + +struct TLSContext *tls_import_context(const unsigned char *buffer, unsigned int buf_len) { + if ((!buffer) || (buf_len < 64) || (buffer[0] != TLS_SERIALIZED_OBJECT) || (buffer[5] != 0x01)) { + DEBUG_PRINT("CANNOT IMPORT CONTEXT BUFFER\n"); + return NULL; + } + // create a context object + struct TLSContext *context = tls_create_context(0, TLS_V12); + if (context) { + unsigned char temp[0xFF]; + context->version = ntohs(*(unsigned short *)&buffer[1]); + unsigned short length = ntohs(*(unsigned short *)&buffer[3]); + if (length != buf_len - 5) { + DEBUG_PRINT("INVALID IMPORT BUFFER SIZE\n"); + tls_destroy_context(context); + return NULL; + } + context->connection_status = buffer[6]; + context->cipher = ntohs(*(unsigned short *)&buffer[7]); + unsigned char server = buffer[9]; + if (server == 2) { + context->is_server = 1; + context->is_child = 1; + } else + context->is_server = server; + + unsigned char local_iv[TLS_AES_IV_LENGTH]; + unsigned char remote_iv[TLS_AES_IV_LENGTH]; + unsigned char iv_len = buffer[10]; + if (iv_len > TLS_AES_IV_LENGTH) { + DEBUG_PRINT("INVALID IV LENGTH\n"); + tls_destroy_context(context); + return NULL; + } + + // get the initialization vectors + int buf_pos = 11; + memcpy(local_iv, &buffer[buf_pos], iv_len); + buf_pos += iv_len; + memcpy(remote_iv, &buffer[buf_pos], iv_len); + buf_pos += iv_len; + + unsigned char key_lengths = buffer[buf_pos++]; + TLS_IMPORT_CHECK_SIZE(buf_pos, key_lengths, buf_len) + memcpy(temp, &buffer[buf_pos], key_lengths); + buf_pos += key_lengths; +#ifdef TLS_REEXPORTABLE + context->exportable = 1; + context->exportable_keys = (unsigned char *)TLS_MALLOC(key_lengths); + memcpy(context->exportable_keys, temp, key_lengths); + context->exportable_size = key_lengths; +#else + context->exportable = 0; +#endif + int is_aead = _private_tls_is_aead(context); +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) { + // ChaCha20 + if (iv_len > TLS_CHACHA20_IV_LENGTH) + iv_len = TLS_CHACHA20_IV_LENGTH; + memcpy(context->crypto.ctx_local_mac.local_nonce, local_iv, iv_len); + memcpy(context->crypto.ctx_remote_mac.remote_nonce, remote_iv, iv_len); + } else +#endif + if (is_aead) { +#ifdef WITH_TLS_13 + if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + if (iv_len > TLS_13_AES_GCM_IV_LENGTH) + iv_len = TLS_13_AES_GCM_IV_LENGTH; + memcpy(context->crypto.ctx_local_mac.local_iv, local_iv, iv_len); + memcpy(context->crypto.ctx_remote_mac.remote_iv, remote_iv, iv_len); + } else { +#endif + if (iv_len > TLS_AES_GCM_IV_LENGTH) + iv_len = TLS_AES_GCM_IV_LENGTH; + memcpy(context->crypto.ctx_local_mac.local_aead_iv, local_iv, iv_len); + memcpy(context->crypto.ctx_remote_mac.remote_aead_iv, remote_iv, iv_len); +#ifdef WITH_TLS_13 + } +#endif + } + if (context->is_server) { + if (_private_tls_crypto_create(context, key_lengths / 2, temp, local_iv, temp + key_lengths / 2, remote_iv)) { + DEBUG_PRINT("ERROR CREATING KEY CONTEXT\n"); + tls_destroy_context(context); + return NULL; + } + } else { + if (_private_tls_crypto_create(context, key_lengths / 2, temp + key_lengths / 2, remote_iv, temp, local_iv)) { + DEBUG_PRINT("ERROR CREATING KEY CONTEXT (CLIENT)\n"); + tls_destroy_context(context); + return NULL; + } + } + memset(temp, 0, sizeof(temp)); + + unsigned char mac_length = buffer[buf_pos++]; + if (mac_length > TLS_MAX_MAC_SIZE) { + DEBUG_PRINT("INVALID MAC SIZE\n"); + tls_destroy_context(context); + return NULL; + } + + if (mac_length) { + TLS_IMPORT_CHECK_SIZE(buf_pos, mac_length, buf_len) + memcpy(context->crypto.ctx_local_mac.local_mac, &buffer[buf_pos], mac_length); + buf_pos += mac_length; + + TLS_IMPORT_CHECK_SIZE(buf_pos, mac_length, buf_len) + memcpy(context->crypto.ctx_remote_mac.remote_mac, &buffer[buf_pos], mac_length); + buf_pos += mac_length; + } else +#ifdef TLS_WITH_CHACHA20_POLY1305 + if (is_aead == 2) { + // ChaCha20 + unsigned int i; + TLS_IMPORT_CHECK_SIZE(buf_pos, 128 + CHACHA_BLOCKLEN * 2, buf_len) + for (i = 0; i < 16; i++) { + context->crypto.ctx_local.chacha_local.input[i] = ntohl(*(unsigned int *)(buffer + buf_pos)); + buf_pos += sizeof(unsigned int); + } + for (i = 0; i < 16; i++) { + context->crypto.ctx_remote.chacha_remote.input[i] = ntohl(*(unsigned int *)(buffer + buf_pos)); + buf_pos += sizeof(unsigned int); + } + memcpy(context->crypto.ctx_local.chacha_local.ks, buffer + buf_pos, CHACHA_BLOCKLEN); + buf_pos += CHACHA_BLOCKLEN; + memcpy(context->crypto.ctx_remote.chacha_remote.ks, buffer + buf_pos, CHACHA_BLOCKLEN); + buf_pos += CHACHA_BLOCKLEN; + } +#endif + + TLS_IMPORT_CHECK_SIZE(buf_pos, 2, buf_len) + unsigned short master_key_len = ntohs(*(unsigned short *)(buffer + buf_pos)); + buf_pos += 2; + if (master_key_len) { + TLS_IMPORT_CHECK_SIZE(buf_pos, master_key_len, buf_len) + context->master_key = (unsigned char *)TLS_MALLOC(master_key_len); + if (context->master_key) { + memcpy(context->master_key, &buffer[buf_pos], master_key_len); + context->master_key_len = master_key_len; + } + buf_pos += master_key_len; + } + + TLS_IMPORT_CHECK_SIZE(buf_pos, 16, buf_len) + + context->local_sequence_number = ntohll(*(uint64_t *)&buffer[buf_pos]); + buf_pos += 8; + context->remote_sequence_number = ntohll(*(uint64_t *)&buffer[buf_pos]); + buf_pos += 8; + + TLS_IMPORT_CHECK_SIZE(buf_pos, 4, buf_len) + unsigned int tls_buffer_len = ntohl(*(unsigned int *)&buffer[buf_pos]); + buf_pos += 4; + TLS_IMPORT_CHECK_SIZE(buf_pos, tls_buffer_len, buf_len) + if (tls_buffer_len) { + context->tls_buffer = (unsigned char *)TLS_MALLOC(tls_buffer_len); + if (context->tls_buffer) { + memcpy(context->tls_buffer, &buffer[buf_pos], tls_buffer_len); + context->tls_buffer_len = tls_buffer_len; + } + buf_pos += tls_buffer_len; + } + + TLS_IMPORT_CHECK_SIZE(buf_pos, 4, buf_len) + unsigned int message_buffer_len = ntohl(*(unsigned int *)&buffer[buf_pos]); + buf_pos += 4; + TLS_IMPORT_CHECK_SIZE(buf_pos, message_buffer_len, buf_len) + if (message_buffer_len) { + context->message_buffer = (unsigned char *)TLS_MALLOC(message_buffer_len); + if (context->message_buffer) { + memcpy(context->message_buffer, &buffer[buf_pos], message_buffer_len); + context->message_buffer_len = message_buffer_len; + } + buf_pos += message_buffer_len; + } + + TLS_IMPORT_CHECK_SIZE(buf_pos, 4, buf_len) + unsigned int application_buffer_len = ntohl(*(unsigned int *)&buffer[buf_pos]); + buf_pos += 4; + context->cipher_spec_set = 1; + TLS_IMPORT_CHECK_SIZE(buf_pos, application_buffer_len, buf_len) + if (application_buffer_len) { + context->application_buffer = (unsigned char *)TLS_MALLOC(application_buffer_len); + if (context->application_buffer) { + memcpy(context->application_buffer, &buffer[buf_pos], application_buffer_len); + context->application_buffer_len = application_buffer_len; + } + buf_pos += application_buffer_len; + } + TLS_IMPORT_CHECK_SIZE(buf_pos, 1, buf_len) + context->dtls = buffer[buf_pos]; + buf_pos++; + if (context->dtls) { + TLS_IMPORT_CHECK_SIZE(buf_pos, 4, buf_len) + context->dtls_epoch_local = ntohs(*(unsigned short *)&buffer[buf_pos]); + buf_pos += 2; + context->dtls_epoch_remote = ntohs(*(unsigned short *)&buffer[buf_pos]); + } + } + return context; +} + +int tls_is_broken(struct TLSContext *context) { + if ((!context) || (context->critical_error)) + return 1; + return 0; +} + +int tls_request_client_certificate(struct TLSContext *context) { + if ((!context) || (!context->is_server)) + return 0; + + context->request_client_certificate = 1; + return 1; +} + +int tls_client_verified(struct TLSContext *context) { + if ((!context) || (context->critical_error)) + return 0; + + return (context->client_verified == 1); +} + +const char *tls_sni(struct TLSContext *context) { + if (!context) + return NULL; + return context->sni; +} + +int tls_sni_nset(struct TLSContext *context, const char *sni, unsigned int len) +{ + if ((!context) || (context->is_server) || (context->critical_error) || (context->connection_status != 0)) + return 0; + TLS_FREE(context->sni); + context->sni = NULL; + if (sni && len > 0) { + context->sni = (char *)TLS_MALLOC(len + 1); + if (context->sni) { + context->sni[len] = 0; + memcpy(context->sni, sni, len); + return 1; + } + } + return 0; +} + +int tls_sni_set(struct TLSContext *context, const char *sni) { + if (!context || !sni) + return 0; + return tls_sni_nset(context, sni, strlen(sni)); +} + +int tls_srtp_set(struct TLSContext *context) { + if ((!context) || (!context->dtls)) + return TLS_GENERIC_ERROR; + context->dtls = 4; + return 0; +} + +int tls_srtp_key(struct TLSContext *context, unsigned char *buffer) { + if ((!context->master_key) || (!context->master_key_len)) + return TLS_GENERIC_ERROR; + + unsigned char material[(SRTP_MASTER_KEY_KEY_LEN + SRTP_MASTER_KEY_SALT_LEN) * 2]; + + if (context->is_server) + _private_tls_prf(context, material, sizeof(material), context->master_key, context->master_key_len, (unsigned char *)"EXTRACTOR-dtls_srtp", 19, context->remote_random, TLS_SERVER_RANDOM_SIZE, context->local_random, TLS_CLIENT_RANDOM_SIZE); + else + _private_tls_prf(context, material, sizeof(material), context->master_key, context->master_key_len, (unsigned char *)"EXTRACTOR-dtls_srtp", 19, context->local_random, TLS_SERVER_RANDOM_SIZE, context->remote_random, TLS_CLIENT_RANDOM_SIZE); + + if (buffer) + memcpy(buffer, material, sizeof(material)); + + DEBUG_DUMP_HEX_LABEL("USING MASTER KEY", context->master_key, context->master_key_len); + return sizeof(material); +} + +int tls_is_stun(const unsigned char *msg, int len) { + if ((!msg) || (len < 20)) + return 0; + + if ((msg[4] != 0x21) || (msg[5] != 0x12) || (msg[6] != 0xa4) || (msg[7] != 0x42)) + return 0; + + return 1; +} + +uint32_t _private_tls_crc32(const unsigned char *s, int n) { + uint32_t crc = 0xFFFFFFFF; + int i; + int j; + + for (i = 0; i < n; i++) { + char ch = s[i]; + for (j = 0; j < 8; j ++) { + uint32_t b = (ch ^ crc) & 1; + crc >>= 1; + if (b) + crc=crc^0xEDB88320; + ch >>= 1; + } + } + return ~crc; +} + +int tls_stun_parse(unsigned char *msg, int len, char *pwd, int pwd_len, unsigned char is_ipv6, unsigned char *addr, unsigned int port, unsigned char *response_buffer) { + // not a stun message? + if ((!msg) || (len < 20)) { + DEBUG_PRINT("INVALID STUN PACKET\n"); + return TLS_GENERIC_ERROR; + } + + if ((msg[4] != 0x21) || (msg[5] != 0x12) || (msg[6] != 0xa4) || (msg[7] != 0x42)) { + DEBUG_PRINT("INVALID STUN PACKET (INVALID MAGIC COOKIE)\n"); + return TLS_GENERIC_ERROR; + } + + int addr_len = 4; + if (is_ipv6) + addr_len = 16; + + unsigned char *stun_message = msg; + + unsigned short type = ntohs(*(unsigned short *)msg); + int msg_len = ntohs(*(unsigned short *)&msg[2]); + + if (msg_len > len - 20) + return -1; + + const unsigned char *magic_cookie = &msg[4]; + const unsigned char *transaction_id = &msg[8]; + + const unsigned char *attributes = &msg[20]; + + switch (type) { + case 0x0001: + break; + case 0x0101: + // ignore + return 0; + default: + DEBUG_PRINT("UNSUPPORTED MESSAGE TYPE %x\n", (int)type); + return TLS_FEATURE_NOT_SUPPORTED; + } + + msg += 20; + + unsigned char hash[20]; + unsigned long hash_len = 20; + + unsigned char md5_hash[16]; + + hmac_state hmac; + hash_state md5_state; + + memset(hash, 0, sizeof(hash)); + int stun_message_len = 20; + + unsigned char secret[16]; + + char key[0x4CE]; + + unsigned char *username = NULL; + int username_len = 0; + + unsigned char *realm = NULL; + int realm_len = 0; + + unsigned char *nonce = NULL; + int nonce_len = 0; + + char *ptr; + + int validated = 0; + + uint32_t priority = 0; + + while (msg_len >= 4) { + unsigned short attr_type = ntohs(*(unsigned short *)msg); + int attr_len = ntohs(*(unsigned short *)&msg[2]); + msg += 4; + msg_len -= 4; + + if (attr_len > msg_len) + return TLS_GENERIC_ERROR; + DEBUG_PRINT("STUN ATTR 0x%04X\n", (int)attr_type); + unsigned short temp; + switch (attr_type) { + case 0x0001: + // MAPPED-ADDRESS + break; + case 0x0006: + // USERNAME + if (attr_len > 513) + return TLS_BROKEN_PACKET; + + username = msg; + username_len = attr_len; + break; + case 0x0008: + // MESSAGE-INTEGRITY + if ((attr_len != 20) || (!username) || (!username_len)) + return -1; + + tls_init(); + + // HMAC is computed on message of size including MESSAGE-INTEGRITY, but not including fingerprint (or other post-MESSAGE-INTEGRITY extensions) + temp = *(unsigned short *)&stun_message[2]; + *(unsigned short *)&stun_message[2] = htons(stun_message_len + attr_len + 4 - 20); + + if ((realm) && (realm_len > 0)) { + ptr = key; + memcpy(ptr, username, username_len); + ptr += username_len; + + *ptr = ':'; + ptr ++; + + if ((realm) && (realm_len > 0)) { + memcpy(ptr, realm, realm_len); + ptr += username_len; + *ptr = ':'; + ptr ++; + } + + memcpy(ptr, pwd, pwd_len); + ptr += pwd_len; + + *ptr = 0; + + + DEBUG_PRINT("KEY: %s\n", key); + + md5_init(&md5_state); + md5_process(&md5_state, (unsigned char *)key, strlen(key)); + md5_done(&md5_state, md5_hash); + + DEBUG_DUMP_HEX_LABEL("HASH", md5_hash, 16); + + hmac_init(&hmac, find_hash("sha1"), md5_hash, 16); + } else + hmac_init(&hmac, find_hash("sha1"), (unsigned char *)pwd, pwd_len); + + + hmac_process(&hmac, stun_message, stun_message_len); + hmac_done(&hmac, hash, &hash_len); + + *(unsigned short *)&stun_message[2] = temp; + + if (memcmp(msg, hash, 16)) { + DEBUG_PRINT("MESSAGE-INTEGRITY check failed\n"); + return TLS_INTEGRITY_FAILED; + } + validated = 1; + break; + case 0x0009: + // ERROR-CODE + break; + case 0x000A: + // UNKNOWN-ATTRIBUTES + break; + case 0x0014: + // REALM + if (attr_len > 763) + return TLS_BROKEN_PACKET; + realm = msg; + realm_len = attr_len; + break; + case 0x0015: + // NONCE + if (attr_len > 763) + return TLS_BROKEN_PACKET; + nonce = msg; + nonce_len = attr_len; + break; + case 0x0020: + // XOR-MAPPED-ADDRESS + break; + case 0x0024: + // PRIORITY + if (attr_len != 4) + return TLS_BROKEN_PACKET; + uint32_t priority; + memcpy(&priority, msg, sizeof(priority)); + priority = ntohl(priority); + break; + } + + while (attr_len % 4) + attr_len ++; + + msg_len -= attr_len; + msg += attr_len; + stun_message_len += attr_len + 4; + } + if (!validated) + return TLS_GENERIC_ERROR; + + if (response_buffer) { + response_buffer[0] = 0x01; + response_buffer[1] = 0x01; + + // size + response_buffer[2] = 0x00; + response_buffer[3] = 0x00; + + response_buffer[4] = 0x21; + response_buffer[5] = 0x12; + response_buffer[6] = 0xa4; + response_buffer[7] = 0x42; + + // transaction ID + memcpy(response_buffer + 8, stun_message + 8, 12); + + // XOR-MAPPED-ADDRESS + response_buffer[20] = 0x00; + response_buffer[21] = 0x20; + + *(unsigned short *)&response_buffer[22] = htons(addr_len + 4); + + response_buffer[24] = 0x00; + if (is_ipv6) + response_buffer[25] = 0x02; + else + response_buffer[25] = 0x01; + + *(unsigned short *)&response_buffer[26] = htons(port); + + response_buffer[26] ^= response_buffer[4]; + response_buffer[27] ^= response_buffer[5]; + + memcpy(response_buffer + 28, addr, addr_len); + + int i; + for (i = 0; i < addr_len; i ++) + response_buffer[28 + i] ^= response_buffer[4 + i % 4]; + + int buffer_index = 28 + addr_len; + + // padding + while (buffer_index % 4) { + response_buffer[buffer_index] = 0x00; + buffer_index ++; + response_buffer[22] ++; + } + + // must be computed before to be included in hmac!!! + *(unsigned short *)&response_buffer[2] = htons(buffer_index + 4); + + hmac_init(&hmac, find_hash("sha1"), (unsigned char *)pwd, pwd_len); + hmac_process(&hmac, response_buffer, buffer_index); + hmac_done(&hmac, response_buffer + buffer_index + 4, &hash_len); + + // hmac + response_buffer[buffer_index] = 0x00; + response_buffer[buffer_index + 1] = 0x08; + response_buffer[buffer_index + 2] = 0x00; + response_buffer[buffer_index + 3] = 0x14; + + buffer_index += 24; + + response_buffer[buffer_index ++] = 0x80; + response_buffer[buffer_index ++] = 0x28; + response_buffer[buffer_index ++] = 0x00; + response_buffer[buffer_index ++] = 0x04; + + *(unsigned short *)&response_buffer[2] = htons(buffer_index - 16); + + uint32_t fingerprint = _private_tls_crc32(response_buffer, buffer_index - 4) ^ 0x5354554e; + *(uint32_t *)&response_buffer[buffer_index] = htonl(fingerprint); + + buffer_index += 4; + + DEBUG_DUMP_HEX_LABEL("STUN RESPONSE>>>>>>>>", response_buffer, buffer_index); + + return buffer_index; + } + return 0; +} + +int tls_stun_build(unsigned char transaction_id[12], char *username, int username_len, char *pwd, int pwd_len, unsigned char *msg) { + if (!msg) + return 0; + + *(unsigned short *)msg = htons(0x0001); + + msg[4] = 0x21; + msg[5] = 0x12; + msg[6] = 0xa4; + msg[7] = 0x42; + + memcpy(msg + 8, transaction_id, 12); + + unsigned char *ptr = msg + 20; + + int len = 20; + if ((username) && (username_len > 0) && (username_len <= 513)) { + *(unsigned short *)&msg[20] = htons(0x0006); + *(unsigned short *)&msg[22] = htons(username_len); + + len += 4; + + memcpy(msg + len, username, username_len); + len += username_len; + + while (len % 4) + msg[len ++] = 0; + } + + *(unsigned short *)&msg[len] = htons(0x0025); + *(unsigned short *)&msg[len + 2] = htons(0); + + len += 4; + + + *(unsigned short *)&msg[len] = htons(0x0008); + *(unsigned short *)&msg[len + 2] = htons(20); + + len += 24; + + *(unsigned short *)&msg[2] = htons(len - 20); + + tls_init(); + + hmac_state hmac; + unsigned long hash_len = 20; + + hmac_init(&hmac, find_hash("sha1"), (unsigned char *)pwd, pwd_len); + hmac_process(&hmac, msg, len - 24); + hmac_done(&hmac, msg + len - 20, &hash_len); + + msg[len ++] = 0x80; + msg[len ++] = 0x28; + msg[len ++] = 0x00; + msg[len ++] = 0x04; + + *(unsigned short *)&msg[2] = htons(len - 16); + + uint32_t fingerprint = _private_tls_crc32(msg, len - 4) ^ 0x5354554e; + *(uint32_t *)&msg[len] = htonl(fingerprint); + + len += 4; + + return len; +} + +int tls_cert_fingerprint(const char *pem_data, int pem_size, char *buffer, unsigned int buf_len) { + unsigned int len = 0; + if ((!buffer) || (!buf_len)) + return TLS_GENERIC_ERROR; + + unsigned char *data = tls_pem_decode((const unsigned char *)pem_data, pem_size, 0, &len); + if (!data) + return TLS_GENERIC_ERROR; + + unsigned char hash[32]; + + hash_state state; + + sha256_init(&state); + sha256_process(&state, data, len); + sha256_done(&state, hash); + + TLS_FREE(data); + + int i; + buffer[0] = 0; + for (i = 0; i < 32; i++) { + if (buf_len <= 1) + break; + if (i) { + snprintf(buffer, buf_len, ":"); + buffer ++; + buf_len --; + } + if (buf_len <= 2) + break; + snprintf(buffer, buf_len, "%02X", (unsigned int)hash[i]); + buffer += 2; + buf_len -= 2; + } + return 0; +} + +int tls_load_root_certificates(struct TLSContext *context, const unsigned char *pem_buffer, int pem_size) { + if (!context) + return TLS_GENERIC_ERROR; + + unsigned int len; + int idx = 0; + + do { + unsigned char *data = tls_pem_decode(pem_buffer, pem_size, idx++, &len); + if ((!data) || (!len)) + break; + struct TLSCertificate *cert = asn1_parse(NULL, data, len, 0); + if (cert) { + if ((cert->version == 2) +#ifdef TLS_X509_V1_SUPPORT + || (cert->version == 0) +#endif + ) { + if (cert->priv) { + DEBUG_PRINT("WARNING - parse error (private key encountered in certificate)\n"); + TLS_FREE(cert->priv); + cert->priv = NULL; + cert->priv_len = 0; + } + context->root_certificates = (struct TLSCertificate **)TLS_REALLOC(context->root_certificates, (context->root_count + 1) * sizeof(struct TLSCertificate *)); + if (!context->root_certificates) { + context->root_count = 0; + return TLS_GENERIC_ERROR; + } + context->root_certificates[context->root_count] = cert; + context->root_count++; + DEBUG_PRINT("Loaded certificate: %i\n", (int)context->root_count); + } else { + DEBUG_PRINT("WARNING - certificate version error (v%i)\n", (int)cert->version); + tls_destroy_certificate(cert); + } + } + TLS_FREE(data); + } while (1); + return context->root_count; +} + +int tls_default_verify(struct TLSContext *context, struct TLSCertificate **certificate_chain, int len) { + int i; + int err; + + if (certificate_chain) { + for (i = 0; i < len; i++) { + struct TLSCertificate *certificate = certificate_chain[i]; + // check validity date + err = tls_certificate_is_valid(certificate); + if (err) + return err; + } + } + // check if chain is valid + err = tls_certificate_chain_is_valid(certificate_chain, len); + if (err) + return err; + + // check certificate subject + if ((!context->is_server) && (context->sni) && (len > 0) && (certificate_chain)) { + err = tls_certificate_valid_subject(certificate_chain[0], context->sni); + if (err) + return err; + } + + err = tls_certificate_chain_is_valid_root(context, certificate_chain, len); + if (err) + return err; + + DEBUG_PRINT("Certificate OK\n"); + return no_error; +} + +int tls_unmake_ktls(struct TLSContext *context, int socket) { +#ifdef WITH_KTLS + struct tls12_crypto_info_aes_gcm_128 crypto_info; + socklen_t crypt_info_size = sizeof(crypto_info); + if (getsockopt(socket, SOL_TLS, TLS_TX, &crypto_info, &crypt_info_size)) { + DEBUG_PRINT("ERROR IN getsockopt\n"); + return TLS_GENERIC_ERROR; + } + memcpy(&context->local_sequence_number, crypto_info.rec_seq, TLS_CIPHER_AES_GCM_128_REC_SEQ_SIZE); + context->local_sequence_number = ntohll(context->local_sequence_number); +#ifdef TLS_RX + crypt_info_size = sizeof(crypto_info); + if (getsockopt(socket, SOL_TLS, TLS_RX, &crypto_info, &crypt_info_size)) { + DEBUG_PRINT("ERROR IN getsockopt\n"); + return TLS_GENERIC_ERROR; + } + memcpy(&context->remote_sequence_number, crypto_info.rec_seq, TLS_CIPHER_AES_GCM_128_REC_SEQ_SIZE); + context->remote_sequence_number = ntohll(context->remote_sequence_number); +#endif + return 0; +#endif + DEBUG_PRINT("TLSe COMPILED WITHOUT kTLS SUPPORT\n"); + return TLS_FEATURE_NOT_SUPPORTED; +} + +int tls_make_ktls(struct TLSContext *context, int socket) { + if ((!context) || (context->critical_error) || (context->connection_status != 0xFF) || (!context->crypto.created)) { + DEBUG_PRINT("CANNOT SWITCH TO kTLS\n"); + return TLS_GENERIC_ERROR; + } + if ((!context->exportable) || (!context->exportable_keys)) { + DEBUG_PRINT("KEY MUST BE EXPORTABLE TO BE ABLE TO USE kTLS\n"); + return TLS_GENERIC_ERROR; + } + if ((context->version != TLS_V12) && (context->version != DTLS_V12) && (context->version != TLS_V13) && (context->version != DTLS_V13)) { + DEBUG_PRINT("kTLS IS SUPPORTED ONLY FOR TLS >= 1.2 AND DTLS >= 1.2\n"); + return TLS_FEATURE_NOT_SUPPORTED; + } + switch (context->cipher) { + case TLS_RSA_WITH_AES_128_GCM_SHA256: + case TLS_DHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: + case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: + case TLS_AES_128_GCM_SHA256: + break; + default: + DEBUG_PRINT("CIPHER UNSUPPORTED: kTLS SUPPORTS ONLY AES 128 GCM CIPHERS\n"); + return TLS_FEATURE_NOT_SUPPORTED; + } +#ifdef WITH_KTLS + if (context->exportable_size < TLS_CIPHER_AES_GCM_128_KEY_SIZE * 2) { + DEBUG_PRINT("INVALID KEY SIZE\n"); + return TLS_GENERIC_ERROR; + } + int err; + struct tls12_crypto_info_aes_gcm_128 crypto_info; + crypto_info.info.cipher_type = TLS_CIPHER_AES_GCM_128; + uint64_t local_sequence_number = htonll(context->local_sequence_number); + + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) { + crypto_info.info.version = TLS_1_2_VERSION; + memcpy(crypto_info.iv, &local_sequence_number, TLS_CIPHER_AES_GCM_128_IV_SIZE); + memcpy(crypto_info.rec_seq, &local_sequence_number, TLS_CIPHER_AES_GCM_128_REC_SEQ_SIZE); + memcpy(crypto_info.key, context->exportable_keys, TLS_CIPHER_AES_GCM_128_KEY_SIZE); + memcpy(crypto_info.salt, context->crypto.ctx_local_mac.local_aead_iv, TLS_CIPHER_AES_GCM_128_SALT_SIZE); + } else if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + crypto_info.info.version = TLS_1_3_VERSION; + memcpy(crypto_info.iv, context->crypto.ctx_local_mac.local_iv + 4, TLS_CIPHER_AES_GCM_128_IV_SIZE); + memcpy(crypto_info.rec_seq, &local_sequence_number, TLS_CIPHER_AES_GCM_128_REC_SEQ_SIZE); + memcpy(crypto_info.key, context->exportable_keys, TLS_CIPHER_AES_GCM_128_KEY_SIZE); + memcpy(crypto_info.salt, context->crypto.ctx_local_mac.local_iv, TLS_CIPHER_AES_GCM_128_SALT_SIZE); + } + + err = setsockopt(socket, SOL_TCP, TCP_ULP, "tls", sizeof("tls")); + if (err) + return err; + +#ifdef TLS_RX + // kernel 4.17 adds TLS_RX support + struct tls12_crypto_info_aes_gcm_128 crypto_info_read; + + crypto_info_read.info.cipher_type = TLS_CIPHER_AES_GCM_128; + + uint64_t remote_sequence_number = htonll(context->remote_sequence_number); + + if ((context->version == TLS_V12) || (context->version == DTLS_V12)) { + crypto_info_read.info.version = TLS_1_2_VERSION; + memcpy(crypto_info_read.iv, &remote_sequence_number, TLS_CIPHER_AES_GCM_128_IV_SIZE); + memcpy(crypto_info_read.rec_seq, &remote_sequence_number, TLS_CIPHER_AES_GCM_128_REC_SEQ_SIZE); + memcpy(crypto_info_read.key, context->exportable_keys + TLS_CIPHER_AES_GCM_128_KEY_SIZE, TLS_CIPHER_AES_GCM_128_KEY_SIZE); + memcpy(crypto_info_read.salt, context->crypto.ctx_remote_mac.remote_aead_iv, TLS_CIPHER_AES_GCM_128_SALT_SIZE); + } else if ((context->version == TLS_V13) || (context->version == DTLS_V13)) { + crypto_info_read.info.version = TLS_1_3_VERSION; + memcpy(crypto_info_read.iv, context->crypto.ctx_remote_mac.remote_iv + 4, TLS_CIPHER_AES_GCM_128_IV_SIZE); + memcpy(crypto_info_read.rec_seq, &remote_sequence_number, TLS_CIPHER_AES_GCM_128_REC_SEQ_SIZE); + memcpy(crypto_info_read.key, context->exportable_keys + TLS_CIPHER_AES_GCM_128_KEY_SIZE, TLS_CIPHER_AES_GCM_128_KEY_SIZE); + memcpy(crypto_info_read.salt, context->crypto.ctx_remote_mac.remote_iv, TLS_CIPHER_AES_GCM_128_SALT_SIZE); + } + + err = setsockopt(socket, SOL_TLS, TLS_RX, &crypto_info_read, sizeof(crypto_info_read)); + if (err) + return err; +#endif + return setsockopt(socket, SOL_TLS, TLS_TX, &crypto_info, sizeof(crypto_info)); +#else + DEBUG_PRINT("TLSe COMPILED WITHOUT kTLS SUPPORT\n"); + return TLS_FEATURE_NOT_SUPPORTED; +#endif +} + +#ifdef DEBUG +void tls_print_certificate(const char *fname) { + unsigned char buf[0xFFFF]; + char out_buf[0xFFFF]; + int size = _private_tls_read_from_file(fname, buf, 0xFFFF); + if (size > 0) { + int idx = 0; + unsigned int len; + do { + unsigned char *data; + if (buf[0] == '-') { + data = tls_pem_decode(buf, size, idx++, &len); + } else { + data = buf; + len = size; + } + if ((!data) || (!len)) + return; + struct TLSCertificate *cert = asn1_parse(NULL, data, len, -1); + if (data != buf) + TLS_FREE(data); + if (cert) { + fprintf(stderr, "%s", tls_certificate_to_string(cert, out_buf, 0xFFFF)); + tls_destroy_certificate(cert); + } + if (data == buf) + break; + } while (1); + } +} +#endif + +int tls_remote_error(struct TLSContext *context) { + if (!context) + return TLS_GENERIC_ERROR; + + return context->error_code; +} + +struct TLSRTCPeerConnection *tls_peerconnection_context(unsigned char active, tls_validation_function certificate_verify, void *userdata) { + const char pwd_chars[] = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_/"; + struct TLSRTCPeerConnection *channel = (struct TLSRTCPeerConnection *)TLS_MALLOC(sizeof(struct TLSRTCPeerConnection)); + if (channel) { + memset(channel, 0, sizeof(struct TLSRTCPeerConnection)); + tls_random(channel->stun_transcation_id, 12); + + unsigned char buffer[32]; + tls_random(buffer, 32); + + int i; + for (i = 0; i < 4; i ++) + channel->local_user[i] = 'A' + buffer[i] % 25; + channel->local_user[4] = 0; + + for (i = 0; i < 24; i ++) + channel->local_pwd[i] = pwd_chars[buffer[i + 4] % (sizeof(pwd_chars) - 1)]; + + channel->local_pwd[24] = 0; + + channel->certificate_verify = certificate_verify; + channel->active = active; + channel->userdata = userdata; + } + + return channel; +} + +struct TLSRTCPeerConnection *tls_peerconnection_duplicate(struct TLSRTCPeerConnection *channel, void *userdata) { + if ((!channel) || (channel->active) || (!channel->context)) + return NULL; + + struct TLSRTCPeerConnection *clone = tls_peerconnection_context(0, channel->certificate_verify, userdata); + clone->context = tls_accept(channel->context); + tls_srtp_set(clone->context); + tls_add_alpn(clone->context, "webrtc"); + + return clone; +} + +struct TLSContext *tls_peerconnection_dtls_context(struct TLSRTCPeerConnection *channel) { + if (!channel) + return NULL; + + return channel->context; +} + +int tls_peerconnection_remote_credentials(struct TLSRTCPeerConnection *channel, char *remote_username, int remote_username_len, char *remote_pwd, int remote_pwd_len, char *remote_fingerprint, int remote_fingerprint_len) { + if (!channel) + return TLS_GENERIC_ERROR; + + if (channel->remote_user) { + TLS_FREE(channel->remote_user); + channel->remote_user = NULL; + channel->remote_user_len = 0; + } + + if (channel->remote_pwd) { + TLS_FREE(channel->remote_pwd); + channel->remote_pwd = NULL; + channel->remote_pwd_len = 0; + } + + if ((remote_username) && (remote_username_len > 0)) { + channel->remote_user = (unsigned char *)TLS_MALLOC(remote_username_len + 1); + if (!channel->remote_user) + return TLS_NO_MEMORY; + + memcpy(channel->remote_user, remote_username, remote_username_len); + channel->remote_user[remote_username_len] = 0; + channel->remote_user_len = remote_username_len; + } + + if ((remote_pwd) && (remote_pwd_len > 0)) { + channel->remote_pwd = (unsigned char *)TLS_MALLOC(remote_pwd_len + 1); + if (!channel->remote_pwd) + return TLS_NO_MEMORY; + + memcpy(channel->remote_pwd, remote_pwd, remote_pwd_len); + channel->remote_pwd[remote_pwd_len] = 0; + channel->remote_pwd_len = remote_pwd_len; + } + + if ((remote_fingerprint) && (remote_fingerprint_len > 0)) { + if (channel->context->dtls_data->remote_fingerprint) + TLS_FREE(channel->context->dtls_data->remote_fingerprint); + + channel->context->dtls_data->remote_fingerprint = (char *)TLS_MALLOC(remote_fingerprint_len + 1); + if (!channel->context->dtls_data->remote_fingerprint) + return TLS_NO_MEMORY; + + memcpy(channel->context->dtls_data->remote_fingerprint, remote_fingerprint, remote_fingerprint_len); + channel->context->dtls_data->remote_fingerprint[remote_fingerprint_len] = 0; + } + + return 0; +} + +const char *tls_peerconnection_local_pwd(struct TLSRTCPeerConnection *channel) { + if (!channel) + return NULL; + + return channel->local_pwd; +} + +const char *tls_peerconnection_local_username(struct TLSRTCPeerConnection *channel) { + if (!channel) + return NULL; + + return channel->local_user; +} + +void *tls_peerconnection_userdata(struct TLSRTCPeerConnection *channel) { + if (!channel) + return NULL; + + return channel->userdata; +} + +int tls_peerconnection_load_keys(struct TLSRTCPeerConnection *channel, const unsigned char *pem_pub_key, int pem_pub_key_size, const unsigned char *pem_priv_key, int pem_priv_key_size) { + if (!channel->context) { + channel->context = tls_create_context(!channel->active, DTLS_V12); + tls_srtp_set(channel->context); + tls_add_alpn(channel->context, "webrtc"); + + if (channel->context->is_server) + channel->context->request_client_certificate = 1; + } + + if (tls_load_certificates(channel->context, pem_pub_key, pem_pub_key_size) < 0) + return TLS_UNSUPPORTED_CERTIFICATE; + + if (tls_load_private_key(channel->context, pem_priv_key, pem_priv_key_size) < 0) + return TLS_UNSUPPORTED_CERTIFICATE; + + return 0; +} + +int _private_tls_peerconnection_buffer_add(struct TLSRTCPeerBuffer **use_buffer, const unsigned char *buf, int len) { + if ((!use_buffer) || (!buf) || (!len)) + return TLS_GENERIC_ERROR; + + struct TLSRTCPeerBuffer *buffer = (struct TLSRTCPeerBuffer *)TLS_MALLOC(sizeof(struct TLSRTCPeerBuffer)); + if (!buffer) + return TLS_NO_MEMORY; + + buffer->buf = (unsigned char *)TLS_MALLOC(len); + if (!buffer->buf) { + TLS_FREE(buffer); + return TLS_NO_MEMORY; + } + + memcpy(buffer->buf, buf, len); + buffer->len = len; + buffer->next = NULL; + + if (*use_buffer) { + struct TLSRTCPeerBuffer *next = *use_buffer; + while ((next) && (next->next)) + next = (struct TLSRTCPeerBuffer *)next->next; + next->next = buffer; + } else + *use_buffer = buffer; + + return len; +} + +int tls_peerconnection_get_write_msg(struct TLSRTCPeerConnection *channel, unsigned char *buf) { + if ((!channel) || (!channel->write_buffer)) + return 0; + + struct TLSRTCPeerBuffer *buffer = channel->write_buffer; + + int len = buffer->len; + + if (!buf) + return len; + + channel->write_buffer = (struct TLSRTCPeerBuffer *)buffer->next; + + memcpy(buf, buffer->buf, buffer->len); + + if (buffer->buf) + TLS_FREE(buffer->buf); + TLS_FREE(buffer); + + return len; +} + +int tls_peerconnection_get_read_msg(struct TLSRTCPeerConnection *channel, unsigned char *buf) { + if ((!channel) || (!channel->read_buffer)) + return 0; + + struct TLSRTCPeerBuffer *buffer = channel->read_buffer; + + int len = buffer->len; + + if (!buf) + return len; + + channel->read_buffer = (struct TLSRTCPeerBuffer *)buffer->next; + + memcpy(buf, buffer->buf, buffer->len); + + if (buffer->buf) + TLS_FREE(buffer->buf); + TLS_FREE(buffer); + + return len; +} + +int tls_peerconnection_connect(struct TLSRTCPeerConnection *channel, tls_peerconnection_write_function write_function) { + if ((!channel) || (!channel->remote_pwd) || (!channel->remote_user)) + return TLS_GENERIC_ERROR; + + unsigned char msg[1024]; + char full_user[1024]; + + snprintf(full_user, sizeof(full_user), "%s:%s", channel->remote_user, channel->local_user); + + int len = tls_stun_build(channel->stun_transcation_id, full_user, strlen(full_user), (char *)channel->remote_pwd, channel->remote_pwd_len, msg); + if (len < 0) + return 0; + + if (!write_function) + return _private_tls_peerconnection_buffer_add(&channel->write_buffer, msg, len); + + return write_function(channel, msg, len); +} + +void _private_dtls_ensure_keys(struct TLSRTCPeerConnection *channel) { +#ifdef TLS_SRTP + if ((channel->remote_state != 3) && (tls_established(channel->context) == 1)) { + DEBUG_PRINT("******** HAVE REMOTE SRTP KEY ***********\n"); + channel->remote_state = 3; + + unsigned char key_buffer[(SRTP_MASTER_KEY_KEY_LEN + SRTP_MASTER_KEY_SALT_LEN) * 2]; + int key_size = tls_srtp_key(channel->context, key_buffer); + + if (key_size > 0) { + DEBUG_DUMP_HEX_LABEL("SRTP KEY", key_buffer, key_size); + + channel->srtp_local = srtp_init(SRTP_AES_CM, SRTP_AUTH_HMAC_SHA1); + channel->srtp_remote = srtp_init(SRTP_AES_CM, SRTP_AUTH_HMAC_SHA1); + + unsigned char *localkey; + unsigned char *remotekey; + unsigned char *localsalt; + unsigned char *remotesalt; + + if (channel->context->is_server) { + remotekey = key_buffer; + localkey = key_buffer + SRTP_MASTER_KEY_KEY_LEN; + remotesalt = key_buffer + SRTP_MASTER_KEY_KEY_LEN + SRTP_MASTER_KEY_KEY_LEN; + localsalt = key_buffer + SRTP_MASTER_KEY_KEY_LEN + SRTP_MASTER_KEY_KEY_LEN + SRTP_MASTER_KEY_SALT_LEN; + } else { + localkey = key_buffer; + remotekey = key_buffer + SRTP_MASTER_KEY_KEY_LEN; + localsalt = key_buffer + SRTP_MASTER_KEY_KEY_LEN + SRTP_MASTER_KEY_KEY_LEN; + remotesalt = key_buffer + SRTP_MASTER_KEY_KEY_LEN + SRTP_MASTER_KEY_KEY_LEN + SRTP_MASTER_KEY_SALT_LEN; + } + + srtp_key(channel->srtp_local, localkey, 16, localsalt, 14, 80); + srtp_key(channel->srtp_remote, remotekey, 16, remotesalt, 14, 80); + } + } +#endif +} + +int tls_peerconnection_iterate(struct TLSRTCPeerConnection *channel, unsigned char *buf, int buf_len, unsigned char *addr, int port, unsigned char is_ipv6, tls_peerconnection_write_function write_function, int *validate_addr) { + if (validate_addr) + *validate_addr = 0; + + if ((!channel) || (!buf) || (buf_len <= 0)) + return 0; + + int err; + struct TLSContext *context = NULL; + if (tls_is_stun(buf, buf_len)) { + DEBUG_PRINT("RECEIVED STUN PACKET\n"); + unsigned char response_buffer[0x8000]; + int len = tls_stun_parse(buf, buf_len, channel->local_pwd, strlen(channel->local_pwd), is_ipv6, addr, port, response_buffer); + + if ((len >= 0) && (validate_addr)) + *validate_addr = 1; + + unsigned short type = ntohs(*(unsigned short *)buf); + + if ((len <= 0) && (type != 0x0101)) + return len; + + if (!channel->remote_state) { + channel->remote_state = 1; + + if (!channel->local_state) + tls_peerconnection_connect(channel, write_function); + } + + if ((type == 0x0101) && (channel->remote_state == 1) && (channel->active)) { + err = tls_client_connect(channel->context); + if (err < 0) + return err; + + context = channel->context; + } + + if (len > 0) { + if (write_function) + err = write_function(channel, response_buffer, len); + else + err = _private_tls_peerconnection_buffer_add(&channel->write_buffer, response_buffer, len); + + if (err <= 0) + return err; + } + context = channel->context; + if (context) { + unsigned int out_buffer_len = 0; + const unsigned char *out_buffer = tls_get_write_buffer(context, &out_buffer_len); + if ((out_buffer) && (out_buffer_len)) { + if (write_function) + err = write_function(channel, out_buffer, out_buffer_len); + else + err = _private_tls_peerconnection_buffer_add(&channel->write_buffer, out_buffer, out_buffer_len); + if (err > 0) + tls_buffer_clear(context); + } + } + + return err; + } + + if ((buf[0] >= 20) && (buf[0] <= 64)) { + DEBUG_PRINT("RECEIVED DTLS PACKET\n"); + if (!channel->remote_state) { + DEBUG_PRINT("NO STUN ASSOCIATION, IGNORED DTLS PACKET\n"); + return 0; + } + if (channel->remote_state == 1) + channel->remote_state = 2; + + err = tls_consume_stream(channel->context, buf, buf_len, channel->certificate_verify); + + unsigned int out_buffer_len = 0; + const unsigned char *out_buffer = tls_get_write_buffer(channel->context, &out_buffer_len); + if ((out_buffer) && (out_buffer_len)) { + if (write_function) + err = write_function(channel, out_buffer, out_buffer_len); + else + err = _private_tls_peerconnection_buffer_add(&channel->write_buffer, out_buffer, out_buffer_len); + if (err > 0) + tls_buffer_clear(channel->context); + } + + if (err < 0) + return err; + + _private_dtls_ensure_keys(channel); + + return 0; + } + + if ((buf[0] >= 128) && (buf[0] <= 191)) { + DEBUG_PRINT("RECEIVED RTP PACKET\n"); +#ifdef TLS_SRTP + if (channel->srtp_remote) { + DEBUG_DUMP_HEX_LABEL("SRTP", buf, buf_len); + if (buf_len > 12) { + unsigned char out[0x4000]; + int out_buffer_len = sizeof(out) - 12; + int len = srtp_decrypt(channel->srtp_remote, 0, buf, 12, buf + 12, buf_len - 12, out + 12, &out_buffer_len); + + if (len >= 0) { + memcpy(out, buf, 12); + DEBUG_DUMP_HEX_LABEL("RTP header", out, 12); + DEBUG_DUMP_HEX_LABEL("RTP payload", out + 12, out_buffer_len); + _private_tls_peerconnection_buffer_add(&channel->read_buffer, out, out_buffer_len + 12); + } + } + } else { + DEBUG_PRINT("DTLS-SRTP HANDSHAKE NOT YET ESTABLISHED\n"); + } +#endif + return 0; + } + return 0; +} + +int tls_peerconnection_status(struct TLSRTCPeerConnection *channel) { + if (!channel) + return -1; + + // 0 not connected + // 1 stun received + // 2 dtls received + // 3 srtp ready + // 4 closed + + int status = channel->remote_state; + if (channel->context->critical_error) + status = 4; + + return status; +} + +void tls_destroy_peerconnection(struct TLSRTCPeerConnection *channel) { + if (!channel) + return; + + if (channel->context) + tls_destroy_context(channel->context); + + if (channel->remote_user) + TLS_FREE(channel->remote_user); + if (channel->remote_pwd) + TLS_FREE(channel->remote_pwd); + +#ifdef TLS_SRTP + if (channel->srtp_local) + srtp_destroy(channel->srtp_local); + if (channel->srtp_remote) + srtp_destroy(channel->srtp_remote); +#endif + + while (channel->write_buffer) { + struct TLSRTCPeerBuffer *next = (struct TLSRTCPeerBuffer *)channel->write_buffer->next;; + if (channel->write_buffer->buf) + TLS_FREE(channel->write_buffer->buf); + TLS_FREE(channel->write_buffer); + channel->write_buffer = next; + } + + while (channel->read_buffer) { + struct TLSRTCPeerBuffer *next = (struct TLSRTCPeerBuffer *)channel->read_buffer->next;; + if (channel->read_buffer->buf) + TLS_FREE(channel->read_buffer->buf); + TLS_FREE(channel->read_buffer); + channel->read_buffer = next; + } + + TLS_FREE(channel); +} + + +#ifdef SSL_COMPATIBLE_INTERFACE + +int SSL_library_init() { + // dummy function + return 1; +} + +void SSL_load_error_strings() { + // dummy function +} + +void OpenSSL_add_all_algorithms() { + // dummy function +} + +void OpenSSL_add_all_ciphers() { + // dummy function +} + +void OpenSSL_add_all_digests() { + // dummy function +} + +void EVP_cleanup() { + // dummy function +} + +int _tls_ssl_private_send_pending(int client_sock, struct TLSContext *context) { + unsigned int out_buffer_len = 0; + const unsigned char *out_buffer = tls_get_write_buffer(context, &out_buffer_len); + unsigned int out_buffer_index = 0; + int send_res = 0; + SOCKET_SEND_CALLBACK write_cb = NULL; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if (ssl_data) + write_cb = (SOCKET_SEND_CALLBACK)ssl_data->send; + while ((out_buffer) && (out_buffer_len > 0)) { + int res; + if (write_cb) + res = write_cb(client_sock, (char *)&out_buffer[out_buffer_index], out_buffer_len, 0); + else + res = send(client_sock, (char *)&out_buffer[out_buffer_index], out_buffer_len, 0); + if (res <= 0) { + if ((!write_cb) && (res < 0)) { +#ifdef _WIN32 + if (WSAGetLastError() == WSAEWOULDBLOCK) { + context->tls_buffer_len = out_buffer_len; + memmove(context->tls_buffer, out_buffer + out_buffer_index, out_buffer_len); + return res; + } +#else + if ((errno == EAGAIN) || (errno == EINTR)) { + context->tls_buffer_len = out_buffer_len; + memmove(context->tls_buffer, out_buffer + out_buffer_index, out_buffer_len); + return res; + } +#endif + } + send_res = res; + break; + } + out_buffer_len -= res; + out_buffer_index += res; + send_res += res; + } + tls_buffer_clear(context); + return send_res; +} + +struct TLSContext *SSL_new(struct TLSContext *context) { + return tls_accept(context); +} + +int SSLv3_server_method() { + return 1; +} + +int SSLv3_client_method() { + return 0; +} + +int SSL_CTX_use_certificate_file(struct TLSContext *context, const char *filename, int dummy) { + // max 64k buffer + unsigned char buf[0xFFFF]; + int size = _private_tls_read_from_file(filename, buf, sizeof(buf)); + if (size > 0) + return tls_load_certificates(context, buf, size); + return size; +} + +int SSL_CTX_use_PrivateKey_file(struct TLSContext *context, const char *filename, int dummy) { + unsigned char buf[0xFFFF]; + int size = _private_tls_read_from_file(filename, buf, sizeof(buf)); + if (size > 0) + return tls_load_private_key(context, buf, size); + + return size; +} + +int SSL_CTX_check_private_key(struct TLSContext *context) { + if ((!context) || (((!context->private_key) || (!context->private_key->der_bytes) || (!context->private_key->der_len)) +#ifdef TLS_ECDSA_SUPPORTED + && ((!context->ec_private_key) || (!context->ec_private_key->der_bytes) || (!context->ec_private_key->der_len)) +#endif + )) + return 0; + return 1; +} + +struct TLSContext *SSL_CTX_new(int method) { +#ifdef WITH_TLS_13 + return tls_create_context(method, TLS_V13); +#else + return tls_create_context(method, TLS_V12); +#endif +} + +void SSL_free(struct TLSContext *context) { + if (context) { + TLS_FREE(context->user_data); + tls_destroy_context(context); + } +} + +void SSL_CTX_free(struct TLSContext *context) { + SSL_free(context); +} + +int SSL_get_error(struct TLSContext *context, int ret) { + if (!context) + return TLS_GENERIC_ERROR; + return context->critical_error; +} + +int SSL_set_fd(struct TLSContext *context, int socket) { + if (!context) + return 0; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if (!ssl_data) { + ssl_data = (SSLUserData *)TLS_MALLOC(sizeof(SSLUserData)); + if (!ssl_data) + return TLS_NO_MEMORY; + memset(ssl_data, 0, sizeof(SSLUserData)); + context->user_data = ssl_data; + } + ssl_data->fd = socket; + return 1; +} + +void *SSL_set_userdata(struct TLSContext *context, void *data) { + if (!context) + return NULL; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if (!ssl_data) { + ssl_data = (SSLUserData *)TLS_MALLOC(sizeof(SSLUserData)); + if (!ssl_data) + return NULL; + memset(ssl_data, 0, sizeof(SSLUserData)); + context->user_data = ssl_data; + } + void *old_data = ssl_data->user_data; + ssl_data->user_data = data; + return old_data; +} + +void *SSL_userdata(struct TLSContext *context) { + if (!context) + return NULL; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if (!ssl_data) + return NULL; + + return ssl_data->user_data; +} + +int SSL_CTX_root_ca(struct TLSContext *context, const char *pem_filename) { + if (!context) + return TLS_GENERIC_ERROR; + + int count = TLS_GENERIC_ERROR; + FILE *f = fopen(pem_filename, "rb"); + if (f) { + fseek(f, 0, SEEK_END); + size_t size = (size_t)ftell(f); + fseek(f, 0, SEEK_SET); + if (size) { + unsigned char *buf = (unsigned char *)TLS_MALLOC(size + 1); + if (buf) { + buf[size] = 1; + if (fread(buf, 1, size, f) == size) { + count = tls_load_root_certificates(context, buf, size); + if (count > 0) { + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if (!ssl_data) { + ssl_data = (SSLUserData *)TLS_MALLOC(sizeof(SSLUserData)); + if (!ssl_data) { + fclose(f); + return TLS_NO_MEMORY; + } + memset(ssl_data, 0, sizeof(SSLUserData)); + context->user_data = ssl_data; + } + if (!ssl_data->certificate_verify) + ssl_data->certificate_verify = tls_default_verify; + } + } + TLS_FREE(buf); + } + } + fclose(f); + } + return count; +} + +void SSL_CTX_set_verify(struct TLSContext *context, int mode, tls_validation_function verify_callback) { + if (!context) + return; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if (!ssl_data) { + ssl_data = (SSLUserData *)TLS_MALLOC(sizeof(SSLUserData)); + if (!ssl_data) + return; + memset(ssl_data, 0, sizeof(SSLUserData)); + context->user_data = ssl_data; + } + if (mode == SSL_VERIFY_NONE) + ssl_data->certificate_verify = NULL; + else + ssl_data->certificate_verify = verify_callback; +} + +int _private_tls_safe_read(struct TLSContext *context, void *buffer, int buf_size) { + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if ((!ssl_data) || (ssl_data->fd < 0)) + return TLS_GENERIC_ERROR; + + SOCKET_RECV_CALLBACK read_cb = (SOCKET_RECV_CALLBACK)ssl_data->recv; + if (read_cb) + return read_cb(ssl_data->fd, (char *)buffer, buf_size, 0); + return recv(ssl_data->fd, (char *)buffer, buf_size, 0); +} + +int SSL_accept(struct TLSContext *context) { + if (!context) + return TLS_GENERIC_ERROR; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if ((!ssl_data) || (ssl_data->fd < 0)) + return TLS_GENERIC_ERROR; + if (tls_established(context)) + return 1; + unsigned char client_message[0xFFFF]; + // accept + int read_size = 0; + while ((read_size = _private_tls_safe_read(context, (char *)client_message, sizeof(client_message))) > 0) { + if (tls_consume_stream(context, client_message, read_size, ssl_data->certificate_verify) >= 0) { + int res = _tls_ssl_private_send_pending(ssl_data->fd, context); + if (res < 0) + return res; + } + if (tls_established(context)) + return 1; + } + if (read_size <= 0) + return TLS_BROKEN_CONNECTION; + return 0; +} + +int SSL_connect(struct TLSContext *context) { + if (!context) + return TLS_GENERIC_ERROR; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if ((!ssl_data) || (ssl_data->fd < 0) || (context->critical_error)) + return TLS_GENERIC_ERROR; + int res = tls_client_connect(context); + if (res < 0) + return res; + res = _tls_ssl_private_send_pending(ssl_data->fd, context); + if (res < 0) + return res; + + int read_size; + unsigned char client_message[0xFFFF]; + + while ((read_size = _private_tls_safe_read(context, (char *)client_message, sizeof(client_message))) > 0) { + if (tls_consume_stream(context, client_message, read_size, ssl_data->certificate_verify) >= 0) { + res = _tls_ssl_private_send_pending(ssl_data->fd, context); + if (res < 0) + return res; + } + if (tls_established(context)) + return 1; + if (context->critical_error) + return TLS_GENERIC_ERROR; + } + return read_size; +} + +int SSL_shutdown(struct TLSContext *context) { + if (!context) + return TLS_GENERIC_ERROR; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if ((!ssl_data) || (ssl_data->fd < 0)) + return TLS_GENERIC_ERROR; + + tls_close_notify(context); + return 0; +} + +int SSL_write(struct TLSContext *context, const void *buf, unsigned int len) { + if (!context) + return TLS_GENERIC_ERROR; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if ((!ssl_data) || (ssl_data->fd < 0)) + return TLS_GENERIC_ERROR; + + int written_size = tls_write(context, (const unsigned char *)buf, len); + if (written_size > 0) { + int res = _tls_ssl_private_send_pending(ssl_data->fd, context); + if (res <= 0) + return res; + } + return written_size; +} + +int SSL_read(struct TLSContext *context, void *buf, unsigned int len) { + if (!context) + return TLS_GENERIC_ERROR; + + if (context->application_buffer_len) + return tls_read(context, (unsigned char *)buf, len); + + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if ((!ssl_data) || (ssl_data->fd < 0) || (context->critical_error)) + return TLS_GENERIC_ERROR; + if (tls_established(context) != 1) + return TLS_GENERIC_ERROR; + + unsigned char client_message[0xFFFF]; + // accept + int read_size; + while ((!context->application_buffer_len) && ((read_size = _private_tls_safe_read(context, (char *)client_message, sizeof(client_message))) > 0)) { + if (tls_consume_stream(context, client_message, read_size, ssl_data->certificate_verify) > 0) + _tls_ssl_private_send_pending(ssl_data->fd, context); + + if ((context->critical_error) && (!context->application_buffer_len)) + return TLS_GENERIC_ERROR; + } + if ((read_size <= 0) && (!context->application_buffer_len)) + return read_size; + + return tls_read(context, (unsigned char *)buf, len); +} + +int SSL_pending(struct TLSContext *context) { + if (!context) + return TLS_GENERIC_ERROR; + return context->application_buffer_len; +} + +int SSL_set_io(struct TLSContext *context, void *recv_cb, void *send_cb) { + if (!context) + return TLS_GENERIC_ERROR; + SSLUserData *ssl_data = (SSLUserData *)context->user_data; + if (!ssl_data) { + ssl_data = (SSLUserData *)TLS_MALLOC(sizeof(SSLUserData)); + if (!ssl_data) + return TLS_NO_MEMORY; + memset(ssl_data, 0, sizeof(SSLUserData)); + context->user_data = ssl_data; + } + ssl_data->recv = recv_cb; + ssl_data->send = send_cb; + return 0; +} +#endif // SSL_COMPATIBLE_INTERFACE + + +#ifdef TLS_SRTP + +struct SRTPContext { + symmetric_CTR aes; + unsigned int salt[4]; + unsigned char mac[TLS_SHA1_MAC_SIZE]; + + symmetric_CTR rtcp_aes; + unsigned int rtcp_salt[4]; + unsigned char rtcp_mac[TLS_SHA1_MAC_SIZE]; + + unsigned int tag_size; + unsigned int roc; + unsigned short seq; + + unsigned char mode; + unsigned char auth_mode; +}; + +struct SRTPContext *srtp_init(unsigned char mode, unsigned char auth_mode) { + struct SRTPContext *context = NULL; + tls_init(); + switch (mode) { + case SRTP_NULL: + break; + case SRTP_AES_CM: + break; + default: + return NULL; + } + + switch (auth_mode) { + case SRTP_AUTH_NULL: + break; + case SRTP_AUTH_HMAC_SHA1: + break; + default: + return NULL; + } + context = (struct SRTPContext *)TLS_MALLOC(sizeof(struct SRTPContext)); + if (context) { + memset(context, 0, sizeof(struct SRTPContext)); + context->mode = mode; + context->auth_mode = auth_mode; + } + return context; +} + +static int _private_tls_srtp_key_derive(const void *key, int keylen, const void *salt, unsigned char label, void *out, int outlen) { + unsigned char iv[16]; + memcpy(iv, salt, 14); + iv[14] = iv[15] = 0; + void *in = TLS_MALLOC(outlen); + if (!in) + return TLS_GENERIC_ERROR; + memset(in, 0, outlen); + + iv[7] ^= label; + + symmetric_CTR aes; + + if (ctr_start(find_cipher("aes"), iv, (const unsigned char *)key, keylen, 0, CTR_COUNTER_BIG_ENDIAN, &aes)) + return TLS_GENERIC_ERROR; + + ctr_encrypt((unsigned char *)in, (unsigned char *)out, outlen, &aes); + TLS_FREE(in); + ctr_done(&aes); + return 0; +} + +int srtp_key(struct SRTPContext *context, const void *key, int keylen, const void *salt, int saltlen, int tag_bits) { + if (!context) + return TLS_GENERIC_ERROR; + if (context->mode == SRTP_AES_CM) { + if ((saltlen < 14) || (keylen < 16)) + return TLS_GENERIC_ERROR; + // key + unsigned char key_buf[16]; + unsigned char iv[16]; + + memset(iv, 0, sizeof(iv)); + + if (_private_tls_srtp_key_derive(key, keylen, salt, 0, key_buf, sizeof(key_buf))) + return TLS_GENERIC_ERROR; + + DEBUG_DUMP_HEX_LABEL("KEY", key_buf, 16) + + if (_private_tls_srtp_key_derive(key, keylen, salt, 1, context->mac, sizeof(context->mac))) + return TLS_GENERIC_ERROR; + + DEBUG_DUMP_HEX_LABEL("AUTH", context->mac, sizeof(context->mac)) + + memset(context->salt, 0, sizeof(context->salt)); + if (_private_tls_srtp_key_derive(key, keylen, salt, 2, context->salt, 14)) + return TLS_GENERIC_ERROR; + + DEBUG_DUMP_HEX_LABEL("SALT", ((unsigned char *)context->salt), 14) + + if (ctr_start(find_cipher("aes"), iv, key_buf, sizeof(key_buf), 0, CTR_COUNTER_BIG_ENDIAN, &context->aes)) + return TLS_GENERIC_ERROR; + + if (_private_tls_srtp_key_derive(key, keylen, salt, 3, key_buf, sizeof(key_buf))) + return TLS_GENERIC_ERROR; + + DEBUG_DUMP_HEX_LABEL("RTCP KEY", key_buf, 16) + + if (_private_tls_srtp_key_derive(key, keylen, salt, 4, context->rtcp_mac, sizeof(context->rtcp_mac))) + return TLS_GENERIC_ERROR; + + DEBUG_DUMP_HEX_LABEL("RTCP AUTH", context->rtcp_mac, sizeof(context->rtcp_mac)) + + memset(context->rtcp_salt, 0, sizeof(context->rtcp_salt)); + if (_private_tls_srtp_key_derive(key, keylen, salt, 5, context->rtcp_salt, 14)) + return TLS_GENERIC_ERROR; + + DEBUG_DUMP_HEX_LABEL("RTCP SALT", ((unsigned char *)context->rtcp_salt), 14) + + memset(iv, 0, sizeof(iv)); + + if (ctr_start(find_cipher("aes"), iv, key_buf, sizeof(key_buf), 0, CTR_COUNTER_BIG_ENDIAN, &context->rtcp_aes)) + return TLS_GENERIC_ERROR; + } + if (context->auth_mode) + context->tag_size = tag_bits / 8; + return 0; +} + +int srtp_inline(struct SRTPContext *context, const char *b64, int tag_bits) { + char out_buffer[1024]; + + if (!b64) + return TLS_GENERIC_ERROR; + + int len = strlen(b64); + if (len >= sizeof(out_buffer)) + len = sizeof(out_buffer); + int size = _private_b64_decode(b64, len, (unsigned char *)out_buffer); + if (size <= 0) + return TLS_GENERIC_ERROR; + switch (context->mode) { + case SRTP_AES_CM: + if (size < 30) + return TLS_BROKEN_PACKET; + return srtp_key(context, out_buffer, 16, out_buffer + 16, 14, tag_bits); + } + return TLS_GENERIC_ERROR; +} + +int srtp_encrypt(struct SRTPContext *context, unsigned char rtcp, const unsigned char *pt_header, int pt_len, const unsigned char *payload, unsigned int payload_len, unsigned char *out, int *out_buffer_len) { + if ((!context) || (!out) || (!out_buffer_len) || (*out_buffer_len < payload_len)) + return TLS_GENERIC_ERROR; + + int out_len = payload_len; + + unsigned short seq = 0; + unsigned int roc = context->roc; + unsigned int ssrc = 0; + + if ((pt_header) && (pt_len >= 12)) { + seq = ntohs(*((unsigned short *)&pt_header[2])); + ssrc = ntohl(*((unsigned long *)&pt_header[8])); + } + + if (seq < context->seq) + roc++; + + unsigned int roc_be = htonl(roc); + if (context->mode) { + if (*out_buffer_len < out_len) + return TLS_NO_MEMORY; + + unsigned int counter[4]; + counter[0] = context->salt[0]; + counter[1] = context->salt[1] ^ htonl (ssrc); + counter[2] = context->salt[2] ^ roc_be; + counter[3] = context->salt[3] ^ htonl (seq << 16); + if (rtcp) { + ctr_setiv((unsigned char *)&counter, 16, &context->rtcp_aes); + if (ctr_encrypt(payload, out, payload_len, &context->rtcp_aes)) + return TLS_GENERIC_ERROR; + } else { + ctr_setiv((unsigned char *)&counter, 16, &context->aes); + if (ctr_encrypt(payload, out, payload_len, &context->aes)) + return TLS_GENERIC_ERROR; + } + } else { + memcpy(out, payload, payload_len); + } + + *out_buffer_len = out_len; + + if (context->auth_mode == SRTP_AUTH_HMAC_SHA1) { + unsigned char digest_out[TLS_SHA1_MAC_SIZE]; + unsigned long dlen = TLS_SHA1_MAC_SIZE; + hmac_state hmac; + int err; + if (rtcp) + err = hmac_init(&hmac, find_hash("sha1"), context->rtcp_mac, 20); + else + err = hmac_init(&hmac, find_hash("sha1"), context->mac, 20); + if (!err) { + if (pt_len) + err = hmac_process(&hmac, pt_header, pt_len); + if (out_len) + err = hmac_process(&hmac, out, payload_len); + err = hmac_process(&hmac, (unsigned char *)&roc_be, 4); + if (!err) + err = hmac_done(&hmac, digest_out, &dlen); + } + if (err) + return TLS_GENERIC_ERROR; + if (dlen > context->tag_size) + dlen = context->tag_size; + + *out_buffer_len += dlen; + memcpy(out + out_len, digest_out, dlen); + } + context->roc = roc; + context->seq = seq; + return 0; +} + +int srtp_decrypt(struct SRTPContext *context, unsigned char rtcp, const unsigned char *pt_header, int pt_len, const unsigned char *payload, unsigned int payload_len, unsigned char *out, int *out_buffer_len) { + if ((!context) || (!out) || (!out_buffer_len) || (*out_buffer_len < payload_len) || (payload_len < context->tag_size) || (!pt_header) || ((pt_len < 12) && (!rtcp)) || ((pt_len < 8) && (rtcp))) + return TLS_GENERIC_ERROR; + + int out_len = payload_len; + + unsigned short seq = ntohs(*((unsigned short *)&pt_header[2])); + unsigned int roc = context->roc; + unsigned int ssrc = rtcp ? ntohl(*((unsigned long *)&pt_header[4])) : ntohl(*((unsigned long *)&pt_header[8])); + + if (seq < context->seq) + roc++; + + unsigned int roc_be = htonl(roc); + if (context->mode) { + unsigned int counter[4]; + counter[0] = context->salt[0]; + counter[1] = context->salt[1] ^ htonl (ssrc); + counter[2] = context->salt[2] ^ roc_be; + if (rtcp) { + uint32_t srtcp_index = ntohl(*(uint32_t *)&payload[payload_len - context->tag_size - 4]) & 0x7FFFFFFF; + counter[3] = context->salt[3] ^ htonl (srtcp_index); + // ((unsigned cscrhar *)payload)[payload_len - context->tag_size - 4] &= 0x7F; + // DEBUG_DUMP_HEX_LABEL("MODIFIED PACKET", payload, payload_len); + ctr_setiv((unsigned char *)&counter, 16, &context->rtcp_aes); + if (payload_len - context->tag_size - 4 < 0) + return TLS_GENERIC_ERROR; + if (ctr_decrypt(payload, out, payload_len - context->tag_size - 4, &context->rtcp_aes)) + return TLS_GENERIC_ERROR; + } else { + counter[3] = context->salt[3] ^ htonl (seq << 16); + ctr_setiv((unsigned char *)&counter, 16, &context->aes); + if (ctr_decrypt(payload, out, payload_len - context->tag_size, &context->aes)) + return TLS_GENERIC_ERROR; + } + + if (context->auth_mode == SRTP_AUTH_HMAC_SHA1) { + unsigned char digest_out[TLS_SHA1_MAC_SIZE]; + unsigned long dlen = TLS_SHA1_MAC_SIZE; + hmac_state hmac; + int err; + if (rtcp) + err = hmac_init(&hmac, find_hash("sha1"), context->rtcp_mac, sizeof(context->rtcp_mac)); + else + err = hmac_init(&hmac, find_hash("sha1"), context->mac, sizeof(context->mac)); + if (!err) { + if (pt_len) + err = hmac_process(&hmac, pt_header, pt_len); + if ((payload_len - context->tag_size) > 0) + err = hmac_process(&hmac, payload, payload_len - context->tag_size); + err = hmac_process(&hmac, (unsigned char *)&roc_be, 4); + if (!err) + err = hmac_done(&hmac, digest_out, &dlen); + } + if (err) + return TLS_GENERIC_ERROR; + if (dlen > context->tag_size) + dlen = context->tag_size; + + if (memcmp(digest_out, payload + payload_len - context->tag_size, dlen)) { + DEBUG_DUMP_HEX_LABEL("SRTP INTEGRITY FAILED (computed)", digest_out, dlen); + DEBUG_DUMP_HEX_LABEL("SRTP INTEGRITY FAILED (expected)", payload + payload_len - context->tag_size, dlen); + return TLS_INTEGRITY_FAILED; + } + } + } else { + memcpy(out, payload, payload_len - context->tag_size); + } + context->seq = seq; + context->roc = roc; + *out_buffer_len = payload_len - context->tag_size; + return 0; +} + +void srtp_destroy(struct SRTPContext *context) { + if (context) { + if (context->mode) { + ctr_done(&context->aes); + ctr_done(&context->rtcp_aes); + } + + TLS_FREE(context); + } +} + +struct SRTPContext *tls_peerconnection_srtp_local(struct TLSRTCPeerConnection *channel) { + if (!channel) + return NULL; + + return channel->srtp_local; +} + +struct SRTPContext *tls_peerconnection_srtp_remote(struct TLSRTCPeerConnection *channel) { + if (!channel) + return NULL; + + return channel->srtp_remote; +} + +int tls_peerconnection_encrypt(struct TLSRTCPeerConnection *channel, unsigned char rtcp, const unsigned char *pt_header, int pt_len, const unsigned char *payload, unsigned int payload_len, unsigned char *out, int *out_buffer_len) { + return srtp_encrypt(channel->srtp_local, rtcp, pt_header, pt_len, payload, payload_len, out, out_buffer_len); +} + +int tls_peerconnection_decrypt(struct TLSRTCPeerConnection *channel, unsigned char rtcp, const unsigned char *pt_header, int pt_len, const unsigned char *payload, unsigned int payload_len, unsigned char *out, int *out_buffer_len) { + return srtp_decrypt(channel->srtp_remote, rtcp, pt_header, pt_len, payload, payload_len, out, out_buffer_len); +} + +#endif // TLS_SRTP + +#endif // TLSE_C + +int main(){} diff --git a/src/tlse/tlse.h b/src/tlse/tlse.h new file mode 100644 index 0000000..8ceaf03 --- /dev/null +++ b/src/tlse/tlse.h @@ -0,0 +1,472 @@ +#ifndef TLSE_H +#define TLSE_H + +// #define DEBUG + +// define TLS_LEGACY_SUPPORT to support TLS 1.1/1.0 (legacy) +// legacy support it will use an additional 272 bytes / context +#ifndef NO_TLS_LEGACY_SUPPORT +#define TLS_LEGACY_SUPPORT +#endif +// SSL_* style blocking APIs +#ifndef NO_SSL_COMPATIBLE_INTERFACE +#define SSL_COMPATIBLE_INTERFACE +#endif +// support ChaCha20/Poly1305 +#if !defined(__BIG_ENDIAN__) && ((!defined(__BYTE_ORDER)) || (__BYTE_ORDER == __LITTLE_ENDIAN)) + // not working on big endian machines + #ifndef NO_TLS_WITH_CHACHA20_POLY1305 + #define TLS_WITH_CHACHA20_POLY1305 + #endif +#endif +#ifndef NO_TLS_13 +#define WITH_TLS_13 +#endif +// support forward secrecy (Diffie-Hellman ephemeral) +#ifndef NO_TLS_FORWARD_SECRECY +#define TLS_FORWARD_SECRECY +#endif +// support client-side ECDHE +#ifndef NO_TLS_CLIENT_ECDHE +#define TLS_CLIENT_ECDHE +#endif +// suport ecdsa +#ifndef NO_TLS_ECDSA_SUPPORTED +#define TLS_ECDSA_SUPPORTED +#endif +// suport ecdsa client-side +#define TLS_CLIENT_ECDSA +// TLS renegotiation is disabled by default (secured or not) +// do not uncomment next line! +// #define TLS_ACCEPT_SECURE_RENEGOTIATION +// basic superficial X509v1 certificate support +#ifndef NO_TLS_X509_V1_SUPPORT +#define TLS_X509_V1_SUPPORT +#endif + +// disable TLS_RSA_WITH_* ciphers +#ifndef NO_TLS_ROBOT_MITIGATION +#define TLS_ROBOT_MITIGATION +#endif + +#define SSL_V30 0x0300 +#define TLS_V10 0x0301 +#define TLS_V11 0x0302 +#define TLS_V12 0x0303 +#define TLS_V13 0x0304 +#define DTLS_V10 0xFEFF +#define DTLS_V12 0xFEFD +#define DTLS_V13 0xFEFC + +#define TLS_NEED_MORE_DATA 0 +#define TLS_GENERIC_ERROR -1 +#define TLS_BROKEN_PACKET -2 +#define TLS_NOT_UNDERSTOOD -3 +#define TLS_NOT_SAFE -4 +#define TLS_NO_COMMON_CIPHER -5 +#define TLS_UNEXPECTED_MESSAGE -6 +#define TLS_CLOSE_CONNECTION -7 +#define TLS_COMPRESSION_NOT_SUPPORTED -8 +#define TLS_NO_MEMORY -9 +#define TLS_NOT_VERIFIED -10 +#define TLS_INTEGRITY_FAILED -11 +#define TLS_ERROR_ALERT -12 +#define TLS_BROKEN_CONNECTION -13 +#define TLS_BAD_CERTIFICATE -14 +#define TLS_UNSUPPORTED_CERTIFICATE -15 +#define TLS_NO_RENEGOTIATION -16 +#define TLS_FEATURE_NOT_SUPPORTED -17 +#define TLS_DECRYPTION_FAILED -20 + +#define TLS_AES_128_GCM_SHA256 0x1301 +#define TLS_AES_256_GCM_SHA384 0x1302 +#define TLS_CHACHA20_POLY1305_SHA256 0x1303 +#define TLS_AES_128_CCM_SHA256 0x1304 +#define TLS_AES_128_CCM_8_SHA256 0x1305 + +#define TLS_RSA_WITH_AES_128_CBC_SHA 0x002F +#define TLS_RSA_WITH_AES_256_CBC_SHA 0x0035 +#define TLS_RSA_WITH_AES_128_CBC_SHA256 0x003C +#define TLS_RSA_WITH_AES_256_CBC_SHA256 0x003D +#define TLS_RSA_WITH_AES_128_GCM_SHA256 0x009C +#define TLS_RSA_WITH_AES_256_GCM_SHA384 0x009D + +// forward secrecy +#define TLS_DHE_RSA_WITH_AES_128_CBC_SHA 0x0033 +#define TLS_DHE_RSA_WITH_AES_256_CBC_SHA 0x0039 +#define TLS_DHE_RSA_WITH_AES_128_CBC_SHA256 0x0067 +#define TLS_DHE_RSA_WITH_AES_256_CBC_SHA256 0x006B +#define TLS_DHE_RSA_WITH_AES_128_GCM_SHA256 0x009E +#define TLS_DHE_RSA_WITH_AES_256_GCM_SHA384 0x009F + +#define TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA 0xC013 +#define TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA 0xC014 +#define TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 0xC027 +#define TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 0xC02F +#define TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 0xC030 + +#define TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA 0xC009 +#define TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA 0xC00A +#define TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 0xC023 +#define TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 0xC024 +#define TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 0xC02B +#define TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 0xC02C + +#define TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256 0xCCA8 +#define TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256 0xCCA9 +#define TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256 0xCCAA + +#define TLS_FALLBACK_SCSV 0x5600 + +#define TLS_UNSUPPORTED_ALGORITHM 0x00 +#define TLS_RSA_SIGN_RSA 0x01 +#define TLS_RSA_SIGN_MD5 0x04 +#define TLS_RSA_SIGN_SHA1 0x05 +#define TLS_RSA_SIGN_SHA256 0x0B +#define TLS_RSA_SIGN_SHA384 0x0C +#define TLS_RSA_SIGN_SHA512 0x0D +#define TLS_ECDSA_SIGN_SHA256 0x0E + +#define TLS_EC_PUBLIC_KEY 0x11 +#define TLS_EC_prime192v1 0x12 +#define TLS_EC_prime192v2 0x13 +#define TLS_EC_prime192v3 0x14 +#define TLS_EC_prime239v1 0x15 +#define TLS_EC_prime239v2 0x16 +#define TLS_EC_prime239v3 0x17 +#define TLS_EC_prime256v1 0x18 +#define TLS_EC_secp224r1 21 +#define TLS_EC_secp256r1 23 +#define TLS_EC_secp384r1 24 +#define TLS_EC_secp521r1 25 + +#define TLS_ALERT_WARNING 0x01 +#define TLS_ALERT_CRITICAL 0x02 + +#ifdef TLS_ROBOT_MITIGATION + #define TLS_CIPHERS_SIZE(n, mitigated) n * 2 +#else + #define TLS_CIPHERS_SIZE(n, mitigated) (n + mitigated) * 2 +#endif + +#define SRTP_AES128_CM_HMAC_SHA1_80 0x0001 +#define SRTP_AES128_CM_HMAC_SHA1_32 0x0002 +#define SRTP_NULL_HMAC_SHA1_80 0x0005 +#define SRTP_NULL_HMAC_SHA1_32 0x0006 +#define SRTP_AEAD_AES_128_GCM 0x0007 +#define SRTP_AEAD_AES_256_GCM 0x0008 + +#define SRTP_NULL 0 +#define SRTP_AES_CM 1 +#define SRTP_AUTH_NULL 0 +#define SRTP_AUTH_HMAC_SHA1 1 + +#ifdef __cplusplus +extern "C" { +#endif + +typedef enum { + close_notify = 0, + unexpected_message = 10, + bad_record_mac = 20, + decryption_failed_RESERVED = 21, + record_overflow = 22, + decompression_failure = 30, + handshake_failure = 40, + no_certificate_RESERVED = 41, + bad_certificate = 42, + unsupported_certificate = 43, + certificate_revoked = 44, + certificate_expired = 45, + certificate_unknown = 46, + illegal_parameter = 47, + unknown_ca = 48, + access_denied = 49, + decode_error = 50, + decrypt_error = 51, + export_restriction_RESERVED = 60, + protocol_version = 70, + insufficient_security = 71, + internal_error = 80, + inappropriate_fallback = 86, + user_canceled = 90, + no_renegotiation = 100, + unsupported_extension = 110, + no_error = 255 +} TLSAlertDescription; + +// forward declarations +struct TLSPacket; +struct TLSCertificate; +struct TLSContext; +struct ECCCurveParameters; +typedef struct TLSContext TLS; +typedef struct TLSCertificate Certificate; +// webrtc datachannel +struct TLSRTCPeerConnection; + +typedef int (*tls_validation_function)(struct TLSContext *context, struct TLSCertificate **certificate_chain, int len); + +/* + Global initialization. Optional, as it will be called automatically; + however, the initialization is not thread-safe, so if you intend to use TLSe + from multiple threads, you'll need to call tls_init() once, from a single thread, + before using the library. + */ +void tls_init(); +unsigned char *tls_pem_decode(const unsigned char *data_in, unsigned int input_length, int cert_index, unsigned int *output_len); +struct TLSCertificate *tls_create_certificate(); +int tls_certificate_valid_subject(struct TLSCertificate *cert, const char *subject); +int tls_certificate_valid_subject_name(const unsigned char *cert_subject, const char *subject); +int tls_certificate_is_valid(struct TLSCertificate *cert); +void tls_certificate_set_copy(unsigned char **member, const unsigned char *val, int len); +void tls_certificate_set_copy_date(unsigned char **member, const unsigned char *val, int len); +void tls_certificate_set_key(struct TLSCertificate *cert, const unsigned char *val, int len); +void tls_certificate_set_priv(struct TLSCertificate *cert, const unsigned char *val, int len); +void tls_certificate_set_sign_key(struct TLSCertificate *cert, const unsigned char *val, int len); +char *tls_certificate_to_string(struct TLSCertificate *cert, char *buffer, int len); +void tls_certificate_set_exponent(struct TLSCertificate *cert, const unsigned char *val, int len); +void tls_certificate_set_serial(struct TLSCertificate *cert, const unsigned char *val, int len); +void tls_certificate_set_algorithm(struct TLSContext *context, unsigned int *algorithm, const unsigned char *val, int len); +void tls_destroy_certificate(struct TLSCertificate *cert); +struct TLSPacket *tls_create_packet(struct TLSContext *context, unsigned char type, unsigned short version, int payload_size_hint); +void tls_destroy_packet(struct TLSPacket *packet); +void tls_packet_update(struct TLSPacket *packet); +int tls_packet_append(struct TLSPacket *packet, const unsigned char *buf, unsigned int len); +int tls_packet_uint8(struct TLSPacket *packet, unsigned char i); +int tls_packet_uint16(struct TLSPacket *packet, unsigned short i); +int tls_packet_uint32(struct TLSPacket *packet, unsigned int i); +int tls_packet_uint24(struct TLSPacket *packet, unsigned int i); +int tls_random(unsigned char *key, int len); + +/* + Get encrypted data to write, if any. Once you've sent all of it, call + tls_buffer_clear(). + */ +const unsigned char *tls_get_write_buffer(struct TLSContext *context, unsigned int *outlen); + +void tls_buffer_clear(struct TLSContext *context); + +/* Returns 1 for established, 0 for not established yet, and -1 for a critical error. */ +int tls_established(struct TLSContext *context); + +/* Discards any unread decrypted data not consumed by tls_read(). */ +void tls_read_clear(struct TLSContext *context); + +/* + Reads any unread decrypted data (see tls_consume_stream). If you don't read all of it, + the remainder will be left in the internal buffers for next tls_read(). Returns -1 for + fatal error, 0 for no more data, or otherwise the number of bytes copied into the buffer + (up to a maximum of the given size). + */ +int tls_read(struct TLSContext *context, unsigned char *buf, unsigned int size); + +struct TLSContext *tls_create_context(unsigned char is_server, unsigned short version); +const struct ECCCurveParameters *tls_set_curve(struct TLSContext *context, const struct ECCCurveParameters *curve); + +/* Create a context for a given client, from a server context. Returns NULL on error. */ +struct TLSContext *tls_accept(struct TLSContext *context); + +int tls_set_default_dhe_pg(struct TLSContext *context, const char *p_hex_str, const char *g_hex_str); +void tls_destroy_context(struct TLSContext *context); +int tls_cipher_supported(struct TLSContext *context, unsigned short cipher); +int tls_cipher_is_fs(struct TLSContext *context, unsigned short cipher); +int tls_choose_cipher(struct TLSContext *context, const unsigned char *buf, int buf_len, int *scsv_set); +int tls_cipher_is_ephemeral(struct TLSContext *context); +const char *tls_cipher_name(struct TLSContext *context); +int tls_is_ecdsa(struct TLSContext *context); +struct TLSPacket *tls_build_client_key_exchange(struct TLSContext *context); +struct TLSPacket *tls_build_server_key_exchange(struct TLSContext *context, int method); +struct TLSPacket *tls_build_hello(struct TLSContext *context, int tls13_downgrade); +struct TLSPacket *tls_certificate_request(struct TLSContext *context); +struct TLSPacket *tls_build_verify_request(struct TLSContext *context); +int tls_parse_hello(struct TLSContext *context, const unsigned char *buf, int buf_len, unsigned int *write_packets, unsigned int *dtls_verified); +int tls_parse_certificate(struct TLSContext *context, const unsigned char *buf, int buf_len, int is_client); +int tls_parse_server_key_exchange(struct TLSContext *context, const unsigned char *buf, int buf_len); +int tls_parse_client_key_exchange(struct TLSContext *context, const unsigned char *buf, int buf_len); +int tls_parse_server_hello_done(struct TLSContext *context, const unsigned char *buf, int buf_len); +int tls_parse_finished(struct TLSContext *context, const unsigned char *buf, int buf_len, unsigned int *write_packets); +int tls_parse_verify(struct TLSContext *context, const unsigned char *buf, int buf_len); +int tls_parse_payload(struct TLSContext *context, const unsigned char *buf, int buf_len, tls_validation_function certificate_verify); +int tls_parse_message(struct TLSContext *context, unsigned char *buf, int buf_len, tls_validation_function certificate_verify); +int tls_certificate_verify_signature(struct TLSCertificate *cert, struct TLSCertificate *parent); +int tls_certificate_chain_is_valid(struct TLSCertificate **certificates, int len); +int tls_certificate_chain_is_valid_root(struct TLSContext *context, struct TLSCertificate **certificates, int len); + +/* + Add a certificate or a certificate chain to the given context, in PEM form. + Returns a negative value (TLS_GENERIC_ERROR etc.) on error, 0 if there were no + certificates in the buffer, or the number of loaded certificates on success. + */ +int tls_load_certificates(struct TLSContext *context, const unsigned char *pem_buffer, int pem_size); + +/* + Add a private key to the given context, in PEM form. Returns a negative value + (TLS_GENERIC_ERROR etc.) on error, 0 if there was no private key in the + buffer, or 1 on success. + */ +int tls_load_private_key(struct TLSContext *context, const unsigned char *pem_buffer, int pem_size); +struct TLSPacket *tls_build_certificate(struct TLSContext *context); +struct TLSPacket *tls_build_finished(struct TLSContext *context); +struct TLSPacket *tls_build_change_cipher_spec(struct TLSContext *context); +struct TLSPacket *tls_build_done(struct TLSContext *context); +struct TLSPacket *tls_build_message(struct TLSContext *context, const unsigned char *data, unsigned int len); +int tls_client_connect(struct TLSContext *context); +int tls_write(struct TLSContext *context, const unsigned char *data, unsigned int len); +struct TLSPacket *tls_build_alert(struct TLSContext *context, char critical, unsigned char code); +int tls_connection_status(struct TLSContext *context); + +/* + Process a given number of input bytes from a socket. If the other side just + presented a certificate and certificate_verify is not NULL, it will be called. + + Returns 0 if there's no data ready yet, a negative value (see + TLS_GENERIC_ERROR etc.) for an error, or a positive value (the number of bytes + used from buf) if one or more complete TLS messages were received. The data + is copied into an internal buffer even if not all of it was consumed, + so you should not re-send it the next time. + + Decrypted data, if any, should be read back with tls_read(). Can change the + status of tls_established(). If the library has anything to send back on the + socket (e.g. as part of the handshake), tls_get_write_buffer() will return + non-NULL. + */ +int tls_consume_stream(struct TLSContext *context, const unsigned char *buf, int buf_len, tls_validation_function certificate_verify); +void tls_close_notify(struct TLSContext *context); +void tls_alert(struct TLSContext *context, unsigned char critical, int code); + +/* Whether tls_consume_stream() has data in its buffer that is not processed yet. */ +int tls_pending(struct TLSContext *context); + +/* + Set the context as serializable or not. Must be called before negotiation. + Exportable contexts use a bit more memory, to be able to hold the keys. + + Note that imported keys are not reexportable unless TLS_REEXPORTABLE is set. + */ +void tls_make_exportable(struct TLSContext *context, unsigned char exportable_flag); + +int tls_export_context(struct TLSContext *context, unsigned char *buffer, unsigned int buf_len, unsigned char small_version); +struct TLSContext *tls_import_context(const unsigned char *buffer, unsigned int buf_len); +int tls_is_broken(struct TLSContext *context); +int tls_request_client_certificate(struct TLSContext *context); +int tls_client_verified(struct TLSContext *context); +const char *tls_sni(struct TLSContext *context); +int tls_sni_set(struct TLSContext *context, const char *sni); +int tls_sni_nset(struct TLSContext *context, const char *sni, unsigned int len); +// set DTLS-SRTP mode for DTLS context +int tls_srtp_set(struct TLSContext *context); +int tls_srtp_key(struct TLSContext *context, unsigned char *buffer); + +int tls_stun_parse(unsigned char *msg, int len, char *pwd, int pwd_len, unsigned char is_ipv6, unsigned char *addr, unsigned int port, unsigned char *response_buffer); +int tls_stun_build(unsigned char transaction_id[12], char *username, int username_len, char *pwd, int pwd_len, unsigned char *msg); +int tls_is_stun(const unsigned char *msg, int len); + +typedef int (*tls_peerconnection_write_function)(struct TLSRTCPeerConnection *channel, const unsigned char *msg, int msg_len); + +struct TLSRTCPeerConnection *tls_peerconnection_context(unsigned char active, tls_validation_function certificate_verify, void *userdata); +struct TLSRTCPeerConnection *tls_peerconnection_duplicate(struct TLSRTCPeerConnection *channel, void *userdata); +struct TLSContext *tls_peerconnection_dtls_context(struct TLSRTCPeerConnection *channel); +int tls_peerconnection_remote_credentials(struct TLSRTCPeerConnection *channel, char *remote_username, int remote_username_len, char *remote_pwd, int remote_pwd_len, char *remote_fingerprint, int remote_fingerprint_len); +const char *tls_peerconnection_local_pwd(struct TLSRTCPeerConnection *channel); +const char *tls_peerconnection_local_username(struct TLSRTCPeerConnection *channel); +void *tls_peerconnection_userdata(struct TLSRTCPeerConnection *channel); +int tls_peerconnection_load_keys(struct TLSRTCPeerConnection *channel, const unsigned char *pem_pub_key, int pem_pub_key_size, const unsigned char *pem_priv_key, int pem_priv_key_size); +int tls_peerconnection_connect(struct TLSRTCPeerConnection *channel, tls_peerconnection_write_function write_function); +int tls_peerconnection_iterate(struct TLSRTCPeerConnection *channel, unsigned char *buf, int buf_len, unsigned char *addr, int port, unsigned char is_ipv6, tls_peerconnection_write_function write_function, int *validate_addr); +int tls_peerconnection_get_write_msg(struct TLSRTCPeerConnection *channel, unsigned char *buf); +int tls_peerconnection_get_read_msg(struct TLSRTCPeerConnection *channel, unsigned char *buf); +int tls_peerconnection_status(struct TLSRTCPeerConnection *channel); +void tls_destroy_peerconnection(struct TLSRTCPeerConnection *channel); + +int tls_cert_fingerprint(const char *pem_data, int len, char *buffer, unsigned int buf_len); +int tls_load_root_certificates(struct TLSContext *context, const unsigned char *pem_buffer, int pem_size); +int tls_default_verify(struct TLSContext *context, struct TLSCertificate **certificate_chain, int len); +void tls_print_certificate(const char *fname); +int tls_add_alpn(struct TLSContext *context, const char *alpn); +int tls_alpn_contains(struct TLSContext *context, const char *alpn, unsigned char alpn_size); +const char *tls_alpn(struct TLSContext *context); +// useful when renewing certificates for servers, without the need to restart the server +int tls_clear_certificates(struct TLSContext *context); +int tls_make_ktls(struct TLSContext *context, int socket); +int tls_unmake_ktls(struct TLSContext *context, int socket); +/* + Creates a new DTLS random cookie secret to be used in HelloVerifyRequest (server-side). + It is recommended to call this function from time to time, to protect against some + DoS attacks. +*/ +void dtls_reset_cookie_secret(); + +int tls_remote_error(struct TLSContext *context); + +#ifdef SSL_COMPATIBLE_INTERFACE + #define SSL_SERVER_RSA_CERT 1 + #define SSL_SERVER_RSA_KEY 2 + typedef struct TLSContext SSL_CTX; + typedef struct TLSContext SSL; + + #define SSL_FILETYPE_PEM 1 + #define SSL_VERIFY_NONE 0 + #define SSL_VERIFY_PEER 1 + #define SSL_VERIFY_FAIL_IF_NO_PEER_CERT 2 + #define SSL_VERIFY_CLIENT_ONCE 3 + + typedef struct { + int fd; + tls_validation_function certificate_verify; + void *recv; + void *send; + void *user_data; + } SSLUserData; + + int SSL_library_init(); + void SSL_load_error_strings(); + void OpenSSL_add_all_algorithms(); + void OpenSSL_add_all_ciphers(); + void OpenSSL_add_all_digests(); + void EVP_cleanup(); + + int SSLv3_server_method(); + int SSLv3_client_method(); + struct TLSContext *SSL_new(struct TLSContext *context); + int SSL_CTX_use_certificate_file(struct TLSContext *context, const char *filename, int dummy); + int SSL_CTX_use_PrivateKey_file(struct TLSContext *context, const char *filename, int dummy); + int SSL_CTX_check_private_key(struct TLSContext *context); + struct TLSContext *SSL_CTX_new(int method); + void SSL_free(struct TLSContext *context); + void SSL_CTX_free(struct TLSContext *context); + int SSL_get_error(struct TLSContext *context, int ret); + int SSL_set_fd(struct TLSContext *context, int socket); + void *SSL_set_userdata(struct TLSContext *context, void *data); + void *SSL_userdata(struct TLSContext *context); + int SSL_CTX_root_ca(struct TLSContext *context, const char *pem_filename); + void SSL_CTX_set_verify(struct TLSContext *context, int mode, tls_validation_function verify_callback); + int SSL_accept(struct TLSContext *context); + int SSL_connect(struct TLSContext *context); + int SSL_shutdown(struct TLSContext *context); + int SSL_write(struct TLSContext *context, const void *buf, unsigned int len); + int SSL_read(struct TLSContext *context, void *buf, unsigned int len); + int SSL_pending(struct TLSContext *context); + int SSL_set_io(struct TLSContext *context, void *recv, void *send); +#endif + +#ifdef TLS_SRTP + struct SRTPContext; + struct SRTPContext *srtp_init(unsigned char mode, unsigned char auth_mode); + int srtp_key(struct SRTPContext *context, const void *key, int keylen, const void *salt, int saltlen, int tag_bits); + int srtp_inline(struct SRTPContext *context, const char *b64, int tag_bits); + int srtp_encrypt(struct SRTPContext *context, unsigned char rtcp, const unsigned char *pt_header, int pt_len, const unsigned char *payload, unsigned int payload_len, unsigned char *out, int *out_buffer_len); + int srtp_decrypt(struct SRTPContext *context, unsigned char rtcp, const unsigned char *pt_header, int pt_len, const unsigned char *payload, unsigned int payload_len, unsigned char *out, int *out_buffer_len); + void srtp_destroy(struct SRTPContext *context); + + struct SRTPContext *tls_peerconnection_srtp_local(struct TLSRTCPeerConnection *channel); + struct SRTPContext *tls_peerconnection_srtp_remote(struct TLSRTCPeerConnection *channel); + int tls_peerconnection_encrypt(struct TLSRTCPeerConnection *channel, unsigned char rtcp, const unsigned char *pt_header, int pt_len, const unsigned char *payload, unsigned int payload_len, unsigned char *out, int *out_buffer_len); + int tls_peerconnection_decrypt(struct TLSRTCPeerConnection *channel, unsigned char rtcp, const unsigned char *pt_header, int pt_len, const unsigned char *payload, unsigned int payload_len, unsigned char *out, int *out_buffer_len); +#endif + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif

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