Estoy tratando de implementar una rutina de punto fijo que implica calcular el valor de Para pequeños que se acerca . La arquitectura de destino es un FPGA. Un problema es que esta función no se presta fácilmente al uso de la expansión de Taylor. Se puede ver que para valores pequeños de x, la pendiente de va al infinito cuando enfoques , por lo tanto, evaluar la función utilizando una serie de potencia implica multiplicar coeficientes enormes con un pequeño . Por lo tanto, este método es numéricamente inestable.
Usando un enfoque iterativo, Newton-Raphson produce la siguiente ecuación iterativa: , donde estamos tratando de aproximarnos . Pero una vez más, ya que es pequeño, igualmente debería ser pequeño para que la solución converja. Como la ecuación implica dividir un número pequeño por otro número pequeño, es probable que la aritmética de punto fijo falle.
Con eso, me gustaría saber cómo implementar una aproximación de valor pequeño para usando aritmética de punto fijo, ya sea usando coeficientes precalculados o métodos iterativos.
Respuestas:
Una rutina que he usado antes (no sé si es una "adecuada" o no) es un enfoque de divide y vencerás.
Comienza con un valor arbitrario superior e inferior (digamos 5 y 0 respectivamente, las raíces cuadradas más altas y más bajas que desea encontrar) y encuentra el punto medio entre ellas. Cuadrar ese valor.
Si el valor al cuadrado es mayor que su objetivo, establezca el valor superior para que sea su valor al cuadrado. Si es más bajo, establezca el valor más bajo.
Repita hasta que el valor cuadrado coincida con su valor de búsqueda, o haya ejecutado suficientes iteraciones para ser tan preciso como desee.
Aquí hay una pequeña versión que he reunido en perl:
Esto, por supuesto, está usando coma flotante, pero podría adaptarse fácilmente al punto fijo. Puede variar la precisión cambiando el límite de iteración. Cada iteración se vuelve un poco más precisa que la anterior.
por ejemplo: - encuentra la raíz cuadrada de 9:
Si hubiera encontrado el valor 3, por supuesto, se habría detenido antes.
Dale suficientes iteraciones y debería ser muy preciso:
fuente
Aquí hay algunas ideas y rutinas del maestro trascendente ascendido / Guru Scott Dattalo aquí .
Eso es, por supuesto, una broma, excepto la parte del guru (¿Guru?). Scott es excelente.
Discusión relevante. 2005 y PIC y algo es C pero puede ser de valor.
Scott otra vez - 2003
Dos maestros !!!
Dattallo y Golovchenko.
Una gama de métodos
fuente
No especificó lo que quiere decir con "valor pequeño" o "aproximación". Entonces, lo que voy a proponer podría no funcionar, pero aquí va.
Lo más fácil sería hacer una tabla de consulta. Esencialmente, una ROM donde el bus de direcciones es el número que desea raíz cuadrada, y la salida de datos es el resultado. Con un solo BRAM, podría hacer un LUT de 9 bits de entrada y 8 bits de salida. Por supuesto, más BRAM te darán una mesa más grande.
(BRAM = El término Xilinx para un bloque RAM, que también se puede usar como ROM. Otros FPGA tienen cosas similares).
Si desea más precisión de la que BRAM le dará, puede hacer una simple interpolación lineal de dos entradas LUT. Por ejemplo, supongamos que desea una entrada de 12 bits, pero solo tiene BRAM para 10 bits. Usted toma los 10 bits superiores de su entrada y busca eso en la LUT. Agregue 1 a esos 10 bits y busque ese valor también. Luego, realiza una interpolación lineal simple entre los dos resultados, utilizando los 2 bits inferiores para indicar la proporción de un valor sobre el otro. Por supuesto, esto solo le dará una aproximación, pero creo que si hace los cálculos, encontrará que podría ser lo suficientemente bueno.
Este método es el menos preciso con números de bajo valor, pero a medida que la entrada va a valores más altos, la precisión aumenta.
Una optimización del método anterior sería utilizar los BRAM como una ROM de doble puerto. De esta forma, puede leer dos valores sin aumentar el número de BRAM utilizados. Esto también le permitirá calcular un SQRT para cada ciclo de reloj, con algunos retrasos en la canalización.
Por cierto, ¡este método también funciona para SINE / COSINE!
fuente
Prueba el siguiente enfoque
x <<= 2
en C) hasta que esté dentro del rango anterior.fuente
Tratarx = ( y+ d)2≈y2+ 2 dy
Entonces deja re= ( x -y2) / 2 y= ( x / y- y) ≫ 1
y después y= y+ d.
Si MSb es n desde la derecha, deje primeroy= 1 ≪ ( n / 2 ) . Converge en <4 iteraciones.
fuente
Prueba: adivinanzas mejoradas para la primera variable
Su número puede ser considerado: A * 2 ^ n
La primera aproximación es entonces: A * 2 ^ (n / 2)
Supongamos que está utilizando un número de 32 bits, con 24 bits utilizados para contener fracciones. Para números> 1:
1. Cuente el número de bits utilizados en la parte entera (N)
2. Reduzca a la mitad este número (N '= N / 2, es decir, desplazado a la derecha 1 bit)
3. Desplace a la derecha el número original por N' : esta es tu primera suposición.
En este formato, el número más pequeño que puede tener es 2 ^ -24. La raíz cuadrada será de aproximadamente 2 ^ -12. Entonces, para números <1:
1. Cuente el número de bits "cero" en la fracción, hasta alcanzar un bit establecido (N)
2. Reduzca a la mitad este número (N '= N / 2, es decir, desplazado a la derecha 1 bit)
3. IZQUIERDA cambiar el número original por el recuento revisado: esta es su primera suposición.
Ejemplo:
0.0000 0000 0000 0000 1 [16 ceros iniciales] se aproxima a: 0.0000 0000 1
Finalmente, si todavía tiene problemas con la pequeña A: ¿puede calcular 1 / A?
Si es así, invierta su número, luego intente usar el algoritmo Inverse Square Root:
x' = 0.5x * (3 - Ax^2)
fuente