Q (числовой формат) - Q (number format)

Q бинарный фиксированная точка числовой формат, где количество дробный биты (и, возможно, количество целое число бит). Например, число Q15 имеет 15 дробных битов; число Q1.14 имеет 1 целочисленный бит и 14 дробных битов. Формат Q часто используется в оборудовании, не имеющем единицы с плавающей запятой, и в приложениях, требующих постоянное разрешение.

Характеристики

Числа в формате Q условно являются числами с фиксированной запятой, то есть они хранятся и обрабатываются как обычные двоичные целые числа со знаком, что позволяет использовать стандартное целочисленное оборудование /ALU выполнять Рациональное число расчеты. Количество целочисленных битов, дробных битов и размер основного слова должны быть выбраны программистом в зависимости от конкретного приложения - выбор программиста вышеизложенного будет зависеть от диапазона и разрешения, необходимого для чисел.

Некоторые архитектуры DSP предлагают встроенную поддержку общих форматов, таких как Q1.15. В этом случае процессор может поддерживать арифметические операции за один шаг, предлагая насыщенность (для сложения и вычитания) и перенормировки (для умножения) в одной инструкции. Большинство стандартных процессоров этого не делают. Если архитектура напрямую не поддерживает конкретный выбранный формат с фиксированной точкой, программисту потребуется явно обрабатывать насыщение и перенормировку с проверкой границ и сдвигом битов.

Есть два противоречивых обозначения фиксированной точки. Оба обозначения записываются как Qм.п, куда:

  • Q означает, что число находится в обозначении формата Q - Инструменты Техаса представление чисел с фиксированной запятой со знаком (Q напоминает стандартный символ для набора рациональное число ).
  • м. (необязательно, предполагается равным нулю или единице) - это количество битов, отведенных для обозначения двух дополнительных целых частей числа, исключая или включая знаковый бит (поэтому, если m не указано, оно принимается равным нулю или единице) .
  • п - это количество битов, используемых для обозначения дробной части числа, то есть количество битов справа от двоичной точки. (Если n = 0, Q числа целые - вырожденный случай).

Одно соглашение включает знаковый бит в значение м,[1][2] а другая конвенция - нет. Выбор условности можно определить, суммируя т + п. Если значение равно размеру регистра, то бит знака включается в значение м. Если он на единицу меньше размера регистра, знаковый бит не включается в значение м.

Кроме того, буква U может быть добавлена ​​к Q для обозначения беззнакового значения, например UQ1.15, указывающего значения от 0,0 до +1,999969482421875 (то есть ).

Знаковые значения Q хранятся в два дополнения формат, как и целые числа со знаком на большинстве процессоров. В дополнении до двух знаковый бит расширяется до размера регистра.

Для данного Qм.п формат, используя м+п контейнер целых чисел со знаком с п дробные биты:

  • его диапазон
  • его разрешение

Для данного UQм.п формат, используя м+п контейнер целочисленных битов без знака с п дробные биты:

  • его диапазон
  • его разрешение

Например, номер формата Q15.1:

  • требует 15 + 1 = 16 бит
  • его диапазон составляет [-214, 214 - 2−1] = [-16384.0, +16383.5] = [0x8000, 0x8001… 0xFFFF, 0x0000, 0x0001… 0x7FFE, 0x7FFF]
  • его разрешение 2−1 = 0.5

В отличие от плавающая точка чисел, разрешение чисел Q останется постоянным во всем диапазоне.

Преобразование

Поплавок к Q

Чтобы преобразовать число из плавающая точка к Qм.п формат:

  1. Умножьте число с плавающей запятой на 2п
  2. Округлить до ближайшего целого числа

Q плавать

Чтобы преобразовать число из Qм.п формат с плавающей запятой:

  1. Преобразуйте число в число с плавающей запятой, как если бы оно было целым числом, другими словами, удалите двоичную точку
  2. Умножить на 2п

Математические операции

Q числа представляют собой соотношение двух целых чисел: числитель хранится в памяти, знаменатель равен 2п.

Рассмотрим следующий пример:

  • Знаменатель Q8 равен 28 = 256
  • 1,5 равно 384/256
  • 384 сохраняется, 256 выводится, потому что это число Q8.

Если необходимо сохранить базу числа Q (п остается постоянным) математические операции числа Q должны поддерживать постоянный знаменатель. Следующие формулы показывают математические операции с общими числами Q и .

Поскольку знаменатель представляет собой степень двойки, умножение может быть реализовано как арифметический сдвиг влево и деление арифметическим сдвигом вправо; на многих процессорах сдвиги быстрее, чем умножение и деление.

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

С помощью C операции (обратите внимание, что здесь Q относится к количеству бит дробной части):

Добавление

int16_t q_add(int16_t а, int16_t б){    возвращаться а + б;}

С насыщенностью

int16_t q_add_sat(int16_t а, int16_t б){    int16_t результат;    int32_t tmp;    tmp = (int32_t)а + (int32_t)б;    если (tmp > 0x7FFF)        tmp = 0x7FFF;    если (tmp < -1 * 0x8000)        tmp = -1 * 0x8000;    результат = (int16_t)tmp;    возвращаться результат;}

В отличие от чисел с плавающей запятой ± Inf, насыщенные результаты не являются липкими и будут ненасыщенными при добавлении отрицательного значения к положительному значению насыщения (0x7FFF) и наоборот в показанной реализации. В ассемблере можно использовать флаг Signed Overflow, чтобы избежать приведений типов, необходимых для этой реализации C.

Вычитание

int16_t q_sub(int16_t а, int16_t б){    возвращаться а - б;}

Умножение

// предварительно вычисленное значение:#define K (1 << (Q - 1)) // насыщаем до диапазона int16_tint16_t сб16(int32_t Икс){	если (Икс > 0x7FFF) возвращаться 0x7FFF;	еще если (Икс < -0x8000) возвращаться -0x8000;	еще возвращаться (int16_t)Икс;}int16_t q_mul(int16_t а, int16_t б){    int16_t результат;    int32_t темп;    темп = (int32_t)а * (int32_t)б; // тип результата - тип операнда    // Округление; средние значения округляются в большую сторону    темп += K;    // Корректируем путем деления на базу и насыщаем результат    результат = сб16(темп >> Q);    возвращаться результат;}

Разделение

int16_t q_div(int16_t а, int16_t б){    / * предварительное умножение на основание (масштабирование до Q16, чтобы результат был в формате Q8) * /    int32_t темп = (int32_t)а << Q;    / * Округление: средние значения округляются в большую сторону (в меньшую сторону для отрицательных значений). * /    / * ИЛИ сравниваем наиболее значимые биты, т.е. if (((temp >> 31) & 1) == ((b >> 15) & 1)) * /    если ((темп >= 0 && б >= 0) || (темп < 0 && б < 0)) {           темп += б / 2;    / * ИЛИ сдвигает 1 бит, т.е. temp + = (b >> 1); * /    } еще {        темп -= б / 2;    / * ИЛИ сдвигает 1 бит, т.е. temp - = (b >> 1); * /    }    возвращаться (int16_t)(темп / б);}

Смотрите также

Рекомендации

  1. ^ "Руководство по отладчикам ARM Developer Suite и AXD". 1.2. ARM Limited. 2001 [1999]. Глава 4.7.9. AXD> Средства AXD> Форматирование данных> Q-формат. ARM DUI 0066D. В архиве из оригинала от 04.11.2017.
  2. ^ «Глава 4.7.9. AXD> Средства AXD> Форматирование данных> Q-формат». Руководство по отладчикам RealView Development Suite AXD и armd (PDF). 3.0. ARM Limited. 2006 [1999]. С. 4–24. ARM DUI 0066G. В архиве (PDF) из оригинала от 04.11.2017.

дальнейшее чтение

внешняя ссылка