Можно ли уместить игру Minecraft всего в один QR-код?

Ответ: да! И вот же он:

9088bc13c92647cb94c9dcae4e6828a4.png

Игра запускается, и вы можете перемещаться по миру 64×64x64 при помощи клавиш WASD. Пробелом прыгаем, мышью осматриваемся. Щёлкнув левой кнопкой мыши, можно разрушить блок, а правой — установить землю.

Можно просмотреть QR-код при помощи следующей команды под Linux:

zbarcam -1 --raw -Sbinary> /tmp/m4k &&chmod +x /tmp/m4k  && /tmp/m4k
  • -1: выйти после того, как код будет просканирован

  • --raw: не обрабатывать его как текст

  • --Sbinary: воспользоваться двоичной конфигурацией

3e94e2727bd033589e80cb3f42171f74.png

Проект выложен на GitHub здесь: TheSunCat/Minecraft4k.

Как???

Если коротко: ценой боли, страданий, злого колдовства, архивации, потом ещё доза боли, немного удачи, два года трудов— и всё получится.

Длинный ответ: что ж, это долгая история. Если вы готовы поучиться различным творческим приёмам, известным в программировании игр, а также усвоить текст, местами напоминающий шаманские проклятья (не волнуйтесь, все концепции я буду объяснять в контексте, как только придёт их время)— пристегнитесь и поехали!

Примечание: я очень постарался добиться, чтобы этот текст получился доступным и увлекательным. Всякий раз, когда я описываю крупное изменение (резкое или эволюционное), я оставляю ссылку на файл или коммит, о котором идёт речь.

Цель

Итак, насколько велик QR-код? Истина такова, что QR-коды бывают разного размера. Вам, вероятно, наиболее знакома «версия 1»,21 на 21 пиксель. В нём можно хранить 25 текстовых символов, но всего 17 8-разрядных байт.Разница в том, что текст эффективнее поддаётся кодировке в виде QR-кода, нежели отдельные байты. В одном байте может содержаться 255 значений, а в QR-совместимом текстовом символе — по определению меньше возможных вариантов.

bea82fad2b5cb14c6633855577f0181d.png

Самый крупный из существующих QR-кодов — такой, как приведён в начале этой статьи — это «версия 40», в нём умещается, не поверите, 2953 байта. Посмотрим-ка, сколько QR-кодов максимального размера потребуется для записи звуковой темы «Villager idle» из Minecraft.

Эхм, это 8605 байт. (Почти) целиком заполняется триQR-кода. Мы же возьмёмся втиснуть играбельный вариант Minecraft в объём втрое меньший, чем в данном случае тратится на запись обычного бормотания «хммм», которым встретят вас в Minecraft жители деревни.

Предыстория

В декабре 2009 года создатель Майнкрафт Маркус Перссон выпустил максимально урезанную версию Minecraft для конкурса Java 4k Game Programming Contest. Участники этого конкурса соревновались в разработке игр на Java при условии, что игра должна укладываться в 4 килобайта (4096 байт). Заархивированная версия этой игры доступна на сайте Archive.org. В этой игре мы видим мозаичные пиксельные картинки, отдалённо напоминающие Minecraft. Всё это удалось вместить в… 2581 байт. Ух. Гораздо меньше, чем 4k. ПРИЧЁМ, игра уже в таком виде умещается в QR-коде. Напомню, мы располагаем лимитом в 2953 байта.

Итак, дело сделано, верно? Не вполне! Данная версия работает на основе фреймворка Java Applets, который начал выводиться из употребления в разных браузерах, начиная с 2013 года, а к настоящему времени совершенно нефункционален. Кроме того, думаю, у нас получится лучше. Та версия игры была полна багов, производительность в ней была плохая, а разрешение — низкое. Мало того, чтобы её запустить, нам бы потребовался полноценный браузер с установленной средой Java, в которой был бы активирован плагин Java Applets. Так можно ли в самом деле уместить в 2953 байта полнофункциональную версию Minecraft?

Эра Java

Чтобы добиться прогресса на уровне самой программы, необходимо изменить исходный код. К сожалению, код Minecraft4k в публичный доступ так и не выкладывался и, вероятно, выложен не будет. Все ссылки на посты об этой версии игры приводят на страницу 404, а оригинальная страница по Minecraft4k теперь переадресует нас на домашнюю страницу игры Minecraft на сайте Microsoft. К счастью, не составляет труда извлечь исходный код Java из JAR-файлов! Большинство компилируемых языков, например, C и C++, превращаются в оптимизированный ассемблер, но программы Java компилируются в промежуточный язык, который называется «байт-код Java». Затем виртуальная машина Java интерпретирует этот байт-код в обычный ассемблер, именно это происходит при запуске JAR-файла! Такой байт-код остаётся весьма похожим на исходный код Java, и в нём сохраняется масса дополнительной информации, в частности, имена переменных и функций. Поэтому, воспользовавшись инструментами, авторы которых — настоящие умельцы, можно «декомпилировать» байт-код, сохранённый в Minecraft4k, получив на выходе готовый к использованию исходный код!

Давайте заложим JAR-файл в чудесный декомпилятор jd-gui и…

float f13 = 0.0F;
float f14 = 0.0F;
f14 += (this.M[119] - this.M[115]) * 0.02F;
f13 += (this.M[100] - this.M[97]) * 0.02F;
f4 *= 0.5F;
f5 *= 0.99F;
f6 *= 0.5F;
f4 += f9 * f14 + f10 * f13;
f6 += f10 * f14 - f9 * f13;
f5 += 0.003F;
int m;
label208: for (m = 0; m <3; m++) {
float f16 = f1 + f4 * ((m + 0) % 3 / 2);
float f17 = f2 + f5 * ((m + 1) % 3 / 2);
float f19 = f3 + f6 * ((m + 2) % 3 / 2);
for (int i12 = 0; i12 <12; i12++) {
int i13 = (int)(f16 + (i12 >>0&0x1) * 0.6F - 0.3F) - 64;
int i14 = (int)(f17 + ((i12 >>2) - 1) * 0.8F + 0.65F) - 64;
int i15 = (int)(f19 + (i12 >>1&0x1) * 0.6F - 0.3F) - 64;
if (i13 <0 || i14 <0 || i15 <0 || i13 >= 64 || i14 >= 64 || i15 >= 64 || arrayOfInt2[i13 + i14 * 64 + i15 * 4096] >0) {
if (m != 1)
break label208; 
if (this.M[32] >0&& f5 >0.0F) {
this.M[32] = 0;
                f5 = -0.1F;
break label208;
            } 
            f5 = 0.0F;
break label208;
        } 
    } 
    f1 = f16;
    f2 = f17;
    f3 = f19;
}

Уф. Действительно, Java обычно приберегает при компиляции имена переменных и функций. Но Перссон, по-видимому, отключил эту фичу, так как на эту информацию расходовалось бы место в JAR-файле. У нас есть код, но мы не знаем, что именно он делает. Нам предстоит разобраться в работе каждой отдельной инструкции, тестируя её саму по себе. Давайте займёмся обратной разработкой!

Обратная разработка

Первым делом нужно заставить этот код работать. Пока он по-прежнему опирается на фреймворк Java Applet, поэтому я полностью портировал его для работы с более новым (но, можно сказать, антикварным) фреймворком Java Swing. Таким способом мы откроем в игре окно, в котором отобразятся пиксели. Отлично! Приступаем к обратной разработке кода. Здесь я затрону только основные элементы, так как процесс достаточно длительный.

Наиболее очевидны следующие вещи: 214*128BufferedImage — это, определённо, screen (экран), отрисовываемый в окне. В коде предусмотрен большой цикл, который покадрово обновляет каждый байт на этом экране. Параметры экрана также совпадают с размерами того мозаичного игрового представления, которое мы уже видели.  Массив целых чисел 64×64*64 — это world (игровой мир). Он генерируется при запуске игры, и каждое отдельное значение изменяется в тот момент, когда вы размещаете или разрушаете блок. В игре применяется физика с периодическими фиксированными обновлениями. Это значит, что мы не умножаем движение игрока на длину каждого кадра, а предполагаем, что каждый физический шаг имеет постоянную длину. При этом совершается столько шагов, сколько можно сделать за время, которое длится этот кадр. Польза такого подхода заключается в серьёзном упрощении всех физических расчётов — не приходится приспосабливаться к меняющейся длине шага.

Я задокументировал положение игрока, скорость, направление взгляда, а также ввод, поступающий с мыши и клавиатуры. Мой приятель @JuPaHe64 (вот его твиттер) задокументировал, как именно игра проверяет, допустимое ли движение вы совершили. Если так двигаться нельзя, то игра вас поправляет, не давая вам завязнуть в текстурах. Благодаря этому мы исправили баг, существовавший в оригинальной версии игры — если персонаж прыгал прямо в стену, то он мог в ней застрять.

Ещё за неделю мы с JuPaHe64 задокументировали оставшуюся часть игры. Стараясь максимально уменьшить размер игры, мы наладили взаимодействие двух очень необычных систем.

Атлас текстур

Одной грани блока соответствует отдельная текстура размером 16×16 пикселей. В оригинальной игре одна такая текстура занимает приблизительно 350 байт. В Minecraft4k предусмотрено 6 различных видов блоков, каждый — с тремя уникальными сторонами. Получается 6×3 * 350 = 6300 байт. Даже при архивации в виде JAR-формата (по сути, это тот же ZIP-файл), один блок займёт огромную долю пространства, которым мы располагаем. Как же Перссон решил эту задачу?

В Minecraft4k текстуры не хранятся в растровых представлениях, а генерируются на основе алгоритмов прямо во время выполнения. Ого.

Я ранее никогда такого не видел, но это отличный способ сэкономить пространство. Здесь атлас текстур также генерируется на основе выбранного по умолчанию зерна Java Random:

2cd36559979203ec5076e261cd81db31.png

Оказывается, у такого подхода имеется неожиданное достоинство: совершенно не составляет труда повысить разрешение текстур. Поскольку они генерируются алгоритмически, все узоры сохраняются и при более высокой детализации. Так у нас получилось следующее изображение:

a36f45215ce7680068e77693b1f029a9.png

JuPaHe64 подготовил для него очень добротный комплект текстур — нужно признать, что такой метод действительно удобен, если вам требуется сгенерировать какие-либо текстуры для игр:

3c93a341d4d31c516078b392f35710e4.png

Мне особенно понравились текстуры древесной коры и камней.

Трассировка лучей

Не в этом случае. Нет, отчасти она у нас получилась. Игра Minecraft очень сложна с точки зрения рендеринга, несмотря на то, что графика здесь относительно простая. В реальной игре приходится выполнять массу вычислений, например, для решения следующих задач: не отображать грани блоков, скрытые в настоящий момент, выражать все фигуры в виде треугольников. Также нужно обеспечивать математику, преобразующую все тела из трёхмерного представления в двумерное, которое мы видим на экране. Так мы подбираем максимально простые аналоги технике рендеринга, которая называется растрированием. Все эти сложности слишком велики, учитывая, как мало у нас места, поэтому для их устранения в Minecraft4k применяется особая разновидность трассировки лучей: так называемый маршевый метод (ray marching) или «обратная трассировка вокселей».

Отмечу: воксель — это термин, означающий «трёхмерный пиксель». Именно из вокселей состоит мир Minecraft.

Вот что поэтапно происходит с каждым отрисовываемым пикселем:

1.     Вычисляем направление луча, исходя из координат пикселя и того, куда именно смотрит игрок

2.     Сохраняем исходную позицию игрока

3.     Далее циклически совершаем следующие действия, пока на нашем пути не встретится блок:

  1. 1.     Шаг («марш») вперёд на один блок

    2.2.     Проверка, не упёрся ли луч в твёрдый блок

    4.     Окрашиваем пиксель текстурой, которая соответствует данному блоку.

Как вы уже, вероятно, догадались, реализовать такой алгоритм не составляет труда — следовательно, мы сэкономим много драгоценных байт. Кроме того, выполнив обратную трассировку пикселя, расположенного в центре экрана, можно узнать, на какой блок сейчас смотрит игрок. Так нам не придётся писать отдельную функцию, которая бы получала блок — и мы сэкономим ещё немало места.

В настоящее время трассировка лучей применяется для реализации более сложных эффектов — например, освещения и отражения. JuPaHe64 смог внести в код минимальные изменения и добавить тени, отрисованные с точностью до пикселя. Я же добавил освещение в виде рассеянного света. Сочетая эти приёмы с простым генератором миров, мы смогли сделать несколько интересных снимков, несмотря на низкую производительность:

bc975c921469d146cca93516a82d5570.png7f8b5e0ae4587f7cc7909cceef3687af.png

Правда, возникает проблема: даже без таких изысков, как генерация миров и эффекты, связанные с трассировкой лучей, наша игра на Java сейчас занимает 17757 байт. Это более 6 QR-кодов. Код, который заставил бы эту игру работать на современных системах, использующих Java, просто слишком велик. Мы пойдём другим путём.

Портирование на C++

Давайте попробуем убить сразу двух зайцев и перепишем игру таким образом:

1. Воспользуемся C++. Этот язык я знаю гораздо лучше, так что и дело пойдёт быстрее.  

2. Воспользуемся GPU, чтобы игра разворачивалась в режиме реального времени — дело в том, что на ЦП трассировка лучей может идти очень медленно

Промучившись несколько дней, мне удалось не только портировать новую версию, но и заставить её работать — всё благодаря GPU-ускорению. В этой версии используется вычислительный шейдер из OpenGL. Шейдер — это программа, которая выполняется на GPU и может обрабатывать каждый пиксель отдельно, но все — вместе. Для сравнения: при работе на ЦП нужно дождаться, пока окончательно отобразится конкретный пиксель, и лишь потом можно переходить к рендерингу следующего. На GPU все эти операции происходят одновременно.

Одно из самых интересных изменений мы внесли благодаря присоединившемуся к нам товарищу @HANSHOTFIRST. Тогда мы написали коммит, который озаглавили «bad the shader». Здесь слово «bad» (плохой) — эмоциональное, используется как глагол, так как мы переписали всю игру, а она, видите ли, не заработала. Мы рассудили, что, если сделать её более компактной, то и производительность повысится. Для этого мы решили одновременно обрабатывать все три координатные оси: XYZ. В оригинальной игре трассировка лучей идёт по каждой оси отдельно, то есть, сначала по оси X, затем по оси Y и, наконец, по оси Z. На тот момент мы не видели в этом нужды — и определённо несколько заблуждались, в чём вы можете убедиться на следующем примере:

e51461ca99895a3e0007d500382045b7.png

Потом мы сделали небольшой перерыв и даже портировали игру на Linux, а я тем временем ещё поупражнялся с графическими эффектами, которые может двть трассировка лучей:

bb4d9c08b44bb79320c8dd2b354b2393.png

После этого мне впервые удалось серьёзно ужать размер: я освоил упаковку исполняемых файлов. В инструменте gzexe используется сжатие по модели gzip, и он действительно уменьшает размер исполняемого файла. Также я научился работать с ShaderMinifier. Задача этого инструмента — автоматически уменьшать шейдерный код, удаляя из него комментарии, сокращая имена переменных, избавляясь от ненужных пробелов и переходов на новую строку. Такой подход к шейдерам нужен потому, что они серьёзно отличаются от кода на C++. Код на C++ компилируется в двоичный формат, а шейдеры OpenGL, напротив, требуется хранить в виде исходного кода. Драйверы GPU вычисляют их прямо во время выполнения. Следовательно, все без исключения символы, которые мы хотим сохранить в шейдере, будут сохраняться в виде байтов. Насколько всё это можно уменьшить? На самом деле, получается очень компактно, всего… 11314 байта. Фу ты, чёрт. Это четыре QR-кода. Этот размер требуется уменьшить вчетверо. Возможно ли такое вообще?

Переписать на С? В голове не укладывается

Да, я так и сделал. Переписал игру на C. После долгого перерыва игра родилась заново, на сей раз я её буквально по кускам собирал. Только к августу 2022 года мне удалось довести до ума все базовые составляющие, и получилась вполне функциональная игра. По всем признакам она заслуживает права называться Minecraft и умещается в 4598 байтах. Ух. Это на 1645 байта больше, чем выбранный нами предел, можно сказать, всего чуть более полутора QR-кодов. Внезапно задача стала казаться осуществимой. На сей раз пришлось включить в процесс сборки кое-какие грязные приёмы. До готовности проекта было ещё далеко, но в нём уже начали накапливаться вещи, которые стали меня слегка раздражать. Разберём самые одиозные из них:

-nostartfiles

Мы будем работать с компилятором C, который называется GCC. В нём предусмотрено несколько флагов, передав которые мы, очевидно, сможем уменьшить тот (двоичный) файл, который получим на выходе. С помощью  -s можно удалить отладочную информацию, а при помощи -Os оптимизировать код в сторону уменьшения размера, а не повышения производительности. Правда, здесь есть ещё один аргумент. Все мы знаем вездесущую функцию int main, правда?  Напомню, что это точка входа в любую программу на C и C++, первый код, который в ней выполняется. Сейчас я во всеуслышание расскажу, как мы коллективно обманываемся. На самом деле, точка входа в программу — это void _start, но Великий Компилятор не хочет, чтобы вы это знали. Это часть «великого заговора», направленного на распространение argc и argv. Если совсем серьёзно — на самом деле, программы на C начинаются с функции _start. В ней содержится код для обустройства стека, глобальные переменные, значения argc и argv и различные элементы среды исполнения C. Поскольку без этого можно в основном  обойтись, давайте попробуем просто пропускать main, а вместо неё использовать _start:

void _start() {
// настройка стека
asmvolatile("sub $8, %rsp\n");

// Здесь находится Minecraft 

// выход
asmvolatile(".intel_syntaxnoprefix");
asmvolatile("push 231"); //exit_group
asmvolatile("pop rax");
asmvolatile("xoredi, edi");
asmvolatile("syscall");
asmvolatile(".att_syntax prefix");
    __builtin_unreachable();

vondehi

Инструмент vondehi, подобно gzexe, позволяет уменьшить размер двоичного файла. Для этого он разархивирует этот файл, а затем выполняет. Это своего рода сильно оптимизированный ассемблер, который может предшествовать любому сжатому исполняемому файлу, совместимому с xzcat. В таком виде файл можно будет выполнять и под Linux. К данному моменту ничто даже отдалённо не напоминает нам о поддержке кроссплатформенного кода.

strip

В GNU есть ещё один инструмент, strip при помощи которого можно удалять конкретные разделы из исполняемого файла Linux (в формате Executable Linker Format—ELF). Оказывается, даже если передать компилятору GCC флаг -s, в исполняемом файле остаётся ещё много несущественной информации, и находится она как раз в разделах ELF. Чтобы избавиться от них, воспользуемся strip -R. Я просто исключал их по очереди, до тех пор, пока игра не перестала функционировать. Все эти ухищрения дают нам 4598 байт. Это уже отличный результат, но мы только начали путь к отметке «ниже 2953». Есть несколько очевидных оптимизаций, которые можно внести в шейдерный код. Во-первых, сократим имена однородных переменных (они же «униформные», так называются переменные в составе шейдера, которые не поддаются изменению при помощи ShaderMinifier) и поместим их все в функцию main этого шейдера — так мы обойдёмся без вызовов функций. Всё это позволит нам сэкономить 42 байта. Караул. Как же нам уменьшить код ещё на 1000 с лишним байт? Это большой вопрос, и мне потребовался целый год, чтобы ответить на него. Я оставил Minecraft4k в версии размером 3786 байт, каким он был в сентябре 2022 года и сосредоточился на учёбе.

Последний рывок (sanity--)

На дворе стояло 2 сентября 2023 года. Незадолго до того я вспомнил отличный опыт пользователя MattKC, уместившего в QR-код игру «змейка» . Эта история напомнила мне о Minecraft4k и о том, насколько близко я в самом деле подобрался к финишу. Пришло время совершить последний рывок.

Переходим на тёмную сторону

Значительная доля байт заключена в вызовах к функциям из стандартной библиотеки языка C. sin, cos, fmodf и им подобные. Чтобы избавиться от этих вызовов, мне требовалось реализовать часть этих функций самостоятельно, и они действительно получились меньше, чем простые обращения к функции libc.

// ДОДЕЛАТЬ: довести это до ума или использовать вставки x86 ASM
#define TRIG_PRECISION 20
staticfloatmy_sin(float x)
{
float sine;

float t = x;
float sine = x;
for (int a=1; a < TRIG_PRECISION; ++a)
    {
floatmult = -x*x/((2*a+1)*(2*a));
t *= mult;
sine += t;
    }
return sine;
}

Думаю, вы догадались, как я претворил в реальность комментарий из предыдущего листинга.

floatmy_sin(float x) {
float sine;
asm (
"fsin;"
"fstps %0;"
        : "=m" (sine)
        : "t" (x)
    );
return sine;
}

Напрямую воспользовавшись инструкцией fsin из архитектуры x86, я смог сократить двоичный файл на целых 80 байт.

Выворачивание API

В OpenGL определяется стандартный язык для коммуникации с GPU. На этом языке можете формулировать, что вам нужно. Это отлично, поскольку данный язык (предположительно) будет работать независимо от платформы или конкретного аппаратного обеспечения. Мне уже попадались странные случаи аварийного завершения, случавшиеся с версией Minecraft4k на C++ при работе на GPU от Intel. Они возникали из-за того, что в их реализации для OpenGL данные об игровом мире хранятся совсем не так, как у меня. В каждом драйвере OpenGL хватает своих причуд и багов, поэтому программировать на OpenGL ещё веселее. Добавляется фактор «сюрприза», который очень приятен на таком поле, в остальном полностью самосогласованном. На каком-то компьютере ваш трассировщик лучей может работать, но никогда не знаешь наверняка, заработает ли он на любых компьютерах. Такова природа багов в драйверах OpenGL.

Верно и обратное: некоторые из этих багов нам на руку, поскольку позволяют избавиться от значительной части кода, который считается строго обязательным. На этом мне удалось сэкономить целых 26 байт!

dlsym

Мы не будем просить Linux предоставлять нам функции OpenGL, так как можем вручную загружать и выбирать их. Для этого воспользуемся ещё парой функций: dlopen и dlsym. На этот случай нам потребуется сохранить в виде обычного текста имя каждой функции, которая потребуется нам в бинарнике. На это уходит много байт, но в результате итоговая версия всё-таки укорачивается на 21 байт. Уф.

Но нас это не устраивает

Напомню, что мы архивировали двоичный файл и прикрепляли к нему маленький разархиватор. Поэтому любые улучшения в «сжимаемости» двоичного файла прямо конвертируются для нас в сэкономленные байты. Итак, я открыл двоичный файл в шестнадцатеричном редакторе и начал понемногу отсекать фрагменты. Удивительно, но оказалось, что многие важные на вид части, определённые в спецификации ELF, просто не нужны. Игра продолжает работать, даже если её сильно обкорнать, хотя, не уверен, что так получится с любыми утилитами для Linux ELF. Можете сами посмотреть этот максимально скудный вывод readelf, где почти всё равно нулю:

apm@apg ~/D/m4k ((37570fc8)) >readelf -a Minecraft4k_prepacked.elf
ELF Header:
  Magic:   7f 45 4c 46 00 00 00 00 00 00 00 00 00 00 00 00 
  Class:                             none
  Data:                              none
  Version:                           0
  OS/ABI:                            UNIX - System V
  ABI Version:                       0
  Type:                              EXEC (Executable file)
  Machine:                           Advanced Micro Devices X86-64
  Version:                           0x0
  Entry point address:               0x102ca
  Start of program headers:          0 (bytes into file)
  Start of section headers:          64 (bytes into file)
  Flags:                             0x0
  Size of this header:               0 (bytes)
  Size of program headers:           0 (bytes)
  Number of program headers:         0
  Size of section headers:           0 (bytes)
  Number of section headers:         0
  Section header string table index: 0
readelf: Warning: possibly corrupt ELF file header - it has a non-zero section header offset, but no section headers

There are no section groups in this file.

There are no program headers in this file.

There is no dynamic section in this file.

Кроме того, я отсекаю последние 50 байт несжатого файла и последние 8 байт готового архива. В результате при запуске программы мы получаем забавную новую ошибку: /usr/bin/xzcat: (stdin): Unexpected end of input, но игра всё равно воспроизводится нормально! Я же тем самым сэкономил ещё 28 байт!

Превозмогая рассудок

Не раз переписав разную математику в шейдерном коде, исчёркав от руки множество страниц в попытках упростить отдельные уравнения, а также пойдя на некоторые компромиссы (прощайте, крестик-прицел и функция изменения размеров окна), я довёл истощённый Minecraft4k до размера в 3006 байт. Где же достать ещё 53 байта, которые осталось пустить в расход?

Не считая строк dlsym, самые крупные данные, которые хранятся в этой версии Minecraft4k — это пара констант с плавающей точкой. Структура данных float в несжатом виде занимает четыре байта, но, как указано у InigoQuilez, последние два байта нам зачастую не требуются. Поэтому числа с плавающей точкой поддаются архивации всего до 2 байт каждое. Это же двукратное уменьшение!

Благодарю @b0rk@jvns.ca, подсказавшую мне очень полезный сайт float.exposed. На этом сайте есть отличный интерфейс, через который можно поупражняться в работе с двоичными числами с плавающей точкой. Я смог убедиться, что, если убрать из констант последние два байта, это не слишком сказывается на их значениях. Строго говоря, точность чуть снизилась, но разве игрок заметит, что сила тяжести стала 0.00299072265625 вместо 0.003? Думаю, мы даже не найдём такую аналитическую библиотеку, которая бы это отловила, так что точность вполне нормальная. Применив этот трюков и немного дополнительно настроив флаги архивации, я сократил Minecraft4k до 2981 байта. Оставалось всего 28…

К этому моменту идеи у меня кончились. Я поспрашивал совета у кого только мог, и мне подсказали несколько отличных идей, которые я испробовал. Портировал большие функции на ассемблер и вручную их оптимизировал. Избавился от stdlib, самостоятельно реализовав dlopen и dlsym. Делал линковку под минимальную реализацию musllibc, а не под библиотеку GNU C. Ничто из этого не выгорело. Поэтому я задал себе очень важный вопрос:

Насколько это

Насколько это

хуже этого?

хуже этого?

Скажите, замечаете здесь какой-нибудь подвох? Надеюсь, что нет.

Я удалил тени из-под травы. Извините. Иначе никак не получалось. К счастью, никто этого не заметил и не спросил меня, где они, так что нормаааально.

И на этом работа была закончена! В итоге Minecraft4k уместился в 2952 байта, то есть, получился на байт меньше допустимого предела.

Habrahabr.ru прочитано 1772 раза