HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
matrix.c
Go to the documentation of this file.
1 
23 #include "user_config.h"
24 
25 #include "matrix.h"
26 
29 
36 int TestSquare(mat_t MatA)
37 {
38  return( (MatA.rows == MatA.cols) ? 1 : 0 );
39 }
40 
41 
48 mat_t MatAlloc(int rows, int cols)
49 {
50  int r;
51  mat_t MatA;
52  float *fptr;
53 
54  if(rows < 1)
55  {
56 #if MATDEBUG & 1
57 //FIXME
58  printf("MatAlloc: rows < 1\n");
59 #endif
60  rows = 1;
61  }
62  if(cols< 1)
63  {
64 #if MATDEBUG & 1
65 //FIXME
66  printf("MatAlloc: cols < 1\n");
67 #endif
68  cols = 1;
69  }
70 
71  MatA.data = safecalloc(rows,sizeof(float *));
72  if(MatA.data == NULL)
73  {
74  MatA.rows = 0;
75  MatA.cols = 0;
76  return(MatA);
77  }
78 
79  for (r=0;r<rows;r++)
80  {
81  fptr = safecalloc(cols,sizeof(float));
82  if(fptr == NULL)
83  {
84  MatFree(MatA);
85  return(MatA);
86  }
87 
88  MatA.data[r] = fptr;
89  }
90  MatA.rows = rows;
91  MatA.cols = cols;
92  if(cols == rows)
93  MatA.size = cols;
94  else
95  MatA.size = 0;
96  return(MatA);
97 }
98 
99 
105 MEMSPACE
106 mat_t MatAllocSQ(int size)
107 {
108  return(MatAlloc(size,size));
109 }
110 
111 
116 MEMSPACE
117 void MatFree(mat_t matF)
118 {
119  int i;
120  if(matF.data)
121  {
122  for (i=0;i<matF.rows;i++)
123  {
124  if(matF.data[i])
125  {
126  safefree(matF.data[i]);
127  matF.data[i] = NULL;
128  }
129  else
130  {
131 #if MATDEBUG & 1
132  printf("MatFree: attempt to free null matF row: %d\n", i);
133 #endif
134  }
135  }
136  safefree(matF.data);
137  matF.data = NULL;
138  }
139  else
140  {
141 #if MATDEBUG & 1
142  printf("MatFree: attempt to free null matF\n");
143 #endif
144  }
145 }
146 
147 
153 MEMSPACE
154 mat_t MatLoad(void *V, int rows, int cols)
155 {
156  mat_t MatA = MatAlloc(rows,cols);
157  float *f = (float *) V;
158 
159  int r,c,k;
160 
161  k = 0;
162  for(r=0;r<rows;++r)
163  {
164  for(c=0;c<cols;++c)
165  {
166  MatA.data[r][c] = f[k++];
167  }
168  }
169  return(MatA);
170 }
171 
172 
178 MEMSPACE
179 mat_t MatLoadSQ(void *V, int size)
180 {
181  return(MatLoad(V,size,size));
182 }
183 
184 
189 MEMSPACE
190 void MatPrint(mat_t matrix)
191 {
192  int r,c;
193 
194  printf("size: rows(%d), cols(%d)\n", matrix.rows,matrix.cols);
195  for(r=0;r<matrix.rows;++r)
196  {
197  for(c=0;c<matrix.cols;++c)
198  {
199  printf("%+e ", (double) matrix.data[r][c]);
200  }
201  printf("\n");
202  }
203  printf("\n");
204 }
205 
206 
215 MEMSPACE
216 mat_t DeleteRowCol(mat_t MatA,int row,int col)
217 {
218  int r,c;
219  int rM, cM;
220  mat_t MatM;
221  int rows = MatA.rows;
222  int cols = MatA.cols;
223  if(cols < 2)
224  {
225 //FIXME user error
226  cols = 2;
227 #if MATDEBUG & 1
228  printf("DeleteRowCol cols:%d < 2 error\n",cols);
229 #endif
230  }
231  if(rows < 2)
232  {
233 //FIXME user error
234  rows = 2;
235 #if MATDEBUG & 1
236  printf("DeleteRowCol rows:%d < 2 error\n",rows);
237 #endif
238  }
239 
240  MatM = MatAlloc(rows-1,cols-1);
241  rM = 0;
242  for(r=0;r<MatA.rows;++r)
243  {
244 // delete row
245  if(r == row)
246  continue;
247  cM = 0;
248  for(c=0;c<MatA.cols;++c)
249  {
250 // delete col
251  if(c == col)
252  continue;
253  MatM.data[rM][cM] = MatA.data[r][c];
254  ++cM;
255  }
256  ++rM;
257  }
258  return(MatM);
259 }
260 
261 
267 MEMSPACE
269 {
270  int c,r;
271 // allocate using transposed rows and columns
272  mat_t MatR = MatAlloc(MatA.cols, MatA.rows);
273 // row
274  for (r = 0; r < MatA.rows; r++)
275  {
276 // col
277  for( c = 0 ; c < MatA.cols ; c++ )
278  {
279 // transposed
280  MatR.data[c][r] = MatA.data[r][c];
281  }
282  }
283  return(MatR);
284 }
285 
286 
296 MEMSPACE
297 float Minor(mat_t MatA, int row, int col)
298 {
299  float D;
300  mat_t SubMat = DeleteRowCol(MatA, row, col);
301  D = Determinant(SubMat);
302  MatFree(SubMat);
303  return(D);
304 }
305 
306 
316 MEMSPACE
317 float Cofactor(mat_t MatA, int row, int col)
318 {
319  float D = Minor(MatA, row, col);;
320 // (-1)exp(row+col) is -1 if (row+col) odd otherwise 1.0 if even
321  D *= ( ((row+col) & 1) ? -1.0 : 1.0);
322  return(D);
323 }
324 
325 
332 MEMSPACE
334 {
335  int r,c;
336 // Since the result is the transpose we allocate switching rows and cols here
337  mat_t MatAdj = MatAlloc(MatA.cols,MatA.rows);
338 
339  for(r = 0; r< MatA.rows;++r)
340  {
341  for(c = 0; c < MatA.cols;++c)
342  {
343 // transpose rows and columns when storing Cofactors to get the Adjugate
344  MatAdj.data[c][r] = Cofactor(MatA,r,c);
345  }
346  }
347  return(MatAdj);
348 }
349 
350 
357 MEMSPACE
358 float Determinant(mat_t MatA)
359 {
360  int r,c,n,cc;
361  float D = 0;
362 
363  if(MatA.cols != MatA.rows)
364  {
365 //FIXME error
366 #if MATDEBUG & 1
367  printf("Determinate: Matrix MUST be square!\n");
368 #endif
369  return(D);
370  }
371 
372  if (MatA.size < 1)
373  {
374 #if MATDEBUG & 1
375  printf("Determinate: Matrix size MUST be > 0!\n");
376 #endif
377  return(D);
378  }
379 // 1 x 1 case
380  if (MatA.size == 1)
381  {
382  D = MatA.data[0][0];
383  return(D);
384  }
385 // 2 x 2 case
386  if (MatA.size == 2)
387  {
388  D = MatA.data[0][0] * MatA.data[1][1] - MatA.data[1][0] * MatA.data[0][1];
389  return(D);
390  }
391 // solve > 2 cases by recursive Cofactor method
392  for (n=0;n<MatA.size;++n)
393  {
394  D += MatA.data[0][n] * Cofactor(MatA, 0, n);
395  }
396  return(D);
397 }
398 
399 
407 MEMSPACE
409 {
410  int r,c;
411  float D;
412 
413  mat_t MatAdj;
414 
415 #if MATDEBUG & 2
416  printf("MatA\n");
417  MatPrint(MatA);
418 #endif
419 
420  D=Determinant(MatA);
421 
422  if(!D)
423  {
424 //FIXME flag error somehow
425 #if MATDEBUG & 1
426  printf("Determinant(MatA) = 0!\n\n");
427 #endif
428  return(MatA);
429  }
430 
431 #if MATDEBUG & 2
432  printf("Determinant(MatA):\n%e\n\n", (double) D);
433 #endif
434 
435  MatAdj = Adjugate(MatA);
436 #if MATDEBUG & 2
437  printf("Adjugate(MatA)\n");
438  MatPrint(MatAdj);
439 #endif
440 // row
441  for (r=0;r<MatAdj.rows;++r)
442  {
443 // col
444  for (c=0;c<MatAdj.cols;++c)
445  {
446  MatAdj.data[r][c] /= D;
447  }
448  }
449 #if MATDEBUG & 2
450  printf("Adjugate(MatA)/Determinant(MatA)\n");
451  MatPrint(MatAdj);
452 #endif
453 
454  return(MatAdj);
455 }
456 
457 
465 MEMSPACE
467 {
468 // AT = Transpose(A)
469  mat_t MatAT = Transpose(MatA);
470 
471 // AT * A
472  mat_t MatR = MatMul(MatAT,MatA);
473 
474 // 1/(AT * A)
475  mat_t MatI = Invert(MatR);
476 
477 // Pseudo Inverse (AT � A)�1 � AT\n
478  mat_t MatPI = MatMul(MatI,MatAT);
479 
480  MatFree(MatR);
481  MatFree(MatI);
482  MatFree(MatAT);
483 
484  return(MatPI);
485 }
486 
487 
499 MEMSPACE
500 mat_t MatMul(mat_t MatA, mat_t MatB)
501 {
502  float sum = 0;
503  mat_t MatR = MatAlloc(MatA.rows,MatB.cols);
504 
505  int rA,cB,rB;
506 
507  if (MatA.cols != MatB.rows)
508 #if MATDEBUG & 1
509  printf("error MatA cols(%d) != MatB rows(%d)\n", MatA.cols, MatB.rows);
510 #endif
511 
512 // A row
513  for (rA = 0; rA < MatA.rows; ++rA)
514  {
515 // col B
516  for (cB = 0; cB < MatB.cols; ++cB)
517  {
518 // row B
519  for (rB = 0; rB < MatB.rows; ++rB)
520  {
521  sum += (MatA.data[rA][rB] * MatB.data[rB][cB]);
522  }
523  MatR.data[rA][cB] = sum;
524  sum = 0;
525  }
526  }
527  return(MatR);
528 }
529 
530 
537 MEMSPACE
538 mat_t MatRead(char *name)
539 {
540 
541  FILE *fp;
542  char *ptr;
543  int r,c;
544  int rows = 0;
545  int cols = 0;
546  int cnt;
547  int lines = 0;
548  mat_t MatR;
549  char tmp[128];
550 
551  MatR.rows = 0;
552  MatR.cols = 0;
553  MatR.data = NULL;
554 
555  fp = fopen(name,"rb");
556  if(fp == NULL)
557  {
558  return(MatR);
559  }
560 
561 // Read Matrix header with rows and columns
562  ptr = fgets(tmp,253,fp);
563  if(ptr == NULL)
564  {
565  fclose(fp);
566  return(MatR);
567  }
568 //printf("line:%d, %s\n", lines, tmp);
569  ++lines;
570  cnt = sscanf(ptr,"Matrix R:%d C:%d", (int *) &rows, (int *) &cols);
571  if(rows < 1 || cols < 1)
572  {
573  printf("sscanf: %d\n",cnt);
574  printf("MatRead expected header: Matrix R:%d C:%d\n",tmp);
575  fclose(fp);
576  return(MatR);
577  }
578 
579 // FIXME set limts or just let alloc fail ???
580  MatR = MatAlloc(rows,cols);
581  if(MatR.data == NULL)
582  {
583  printf("MatRead(%s: &d,&d) could not alloc memory\n", name, rows,cols);
584  fclose(fp);
585  return(MatR);
586  }
587 
588  for(r=0;r<rows;++r)
589  {
590 // Read rows and columns
591  ptr = fgets(tmp,253,fp);
592 //printf("line:%d, %s\n", lines, tmp);
593  ++lines;
594  if(ptr == NULL)
595  {
596  fclose(fp);
597  MatFree(MatR);
598  return(MatR);
599  }
600  for(c=0;c<cols;++c)
601  {
602  MatR.data[r][c] = strtod(ptr,&ptr);
603  }
604  }
605  fclose(fp);
606  return(MatR);
607 }
608 
609 
616 MEMSPACE
617 int MatWrite(char *name, mat_t MatW)
618 {
619 
620  FILE *fp;
621  int r,c;
622 
623  fp = fopen(name,"wb");
624  if(fp == NULL)
625  {
626  return(0);
627  }
628 
629  fprintf(fp,"Matrix R:%d C:%d\n",MatW.rows,MatW.cols);
630  for(r=0;r<MatW.rows;++r)
631  {
632  for(c=0;c<MatW.cols;++c)
633  {
634  fprintf(fp,"%e ", (double) MatW.data[r][c]);
635  }
636  fprintf(fp,"\n");
637  }
638  fclose(fp);
639  return(1);
640 }
641 
642 
643 #ifdef MATTEST
644 // =============================================
645 // test1
646 // compute Adj(AJ)
647 
648 float AJ[3][3] =
649 {
650  { -3, 2, -5 },
651  { -1, 0, -2 },
652  { 3, -4, 1 }
653 };
654 
655 // =============================================
656 // test2
657 // Matrix Multiplication
658 // compute C * D
659 
660 float C[3][3] =
661 {
662  { 1,2,0 },
663  { 0, 1, 1 },
664  { 2, 0, 1 }
665 };
666 float D[3][3] =
667 {
668  { 1, 1, 2 },
669  { 2, 1, 1 },
670  { 1, 2, 1 }
671 };
672 
673 // =============================================
674 // test3
675 // Test functions required for 3 point screen calibration
676 // See: https://www.maximintegrated.com/en/app-notes/index.mvp/id/5296
677 // Calibration in touch-screen systems
678 //
679 // Calculation
680 // 1/A * X
681 // 1/A * Y
682 //
683 // Answer should be
684 // xd = 0.0635 x + 0.0024 y + 18.9116
685 // yd = -0.0227 x + 0.1634 y + 37.8887
686 // Where (x, y) are touch panel coordinates
687 // and (xd, yd) is the adjusted screen coordinate
688 float A3[3][3] =
689 {
690  { 650, 2000, 1 },
691  { 2800, 1350, 1 },
692  { 2640, 3500, 1 }
693 };
694 
695 float X3[3][1] =
696 {
697  {65},
698  {200},
699  {195}
700 };
701 
702 float Y3[3][1] =
703 {
704  {350},
705  {195},
706  {550}
707 };
708 
709 // =============================================
710 // test4
711 // Test functions required for 5 point screen calibration
712 // See: https://www.maximintegrated.com/en/app-notes/index.mvp/id/5296
713 // Calibration in touch-screen systems
714 //
715 // Calculation - note: AT = transpose(A)
716 // (1/(AT * A) * AT) * X
717 // (1/(AT * A) * AT) * Y
718 //
719 // Answer should be
720 // xd = 0.0677 x + 0.0190 y - 33.7973
721 // yd = -0.0347 x + 0.2100 y - 27.4030
722 // Where (x, y) are touch panel coordinates
723 // and (xd, yd) is the adjusted screen coordinate
724 
725 float A5[5][3] =
726 {
727  { 1700, 2250, 1 },
728  { 750, 1200, 1 },
729  { 3000, 1500, 1 },
730  { 2500, 3400, 1 },
731  { 600, 3000, 1 }
732 };
733 
734 float X5[5][1] =
735 {
736  {100},
737  {50},
738  {200},
739  {210},
740  {65}
741 };
742 
743 float Y5[5][1] =
744 {
745  {350},
746  {200},
747  {200},
748  {600},
749  {600}
750 };
751 
752 int main(int argc, char *argv[])
753 {
754  mat_t MatA,MatX,MatY;
755  mat_t MatI;
756  mat_t MatPI;
757  mat_t MatC,MatD;
758  mat_t MatR;
759  mat_t MatAdj;
760 
761 // =============================
762 // test Adjugate
763  sep();
764  MatA = MatLoad(AJ,3,3);
765  printf("MatA\n");
766  MatPrint(MatA);
767  MatAdj = Adjugate(MatA);
768  printf("Adjugate(MatA)\n");
769  MatPrint(MatAdj);
770  MatFree(MatAdj);
771  MatFree(MatA);
772  sep();
773  printf("\n");
774 
775 // =============================
776 // test MatMul
777  sep();
778  MatC = MatLoadSQ(C,3);
779  printf("MatC\n");
780  MatPrint(MatC);
781  MatD = MatLoadSQ(D,3);
782  printf("MatD\n");
783  MatPrint(MatD);
784  MatR = MatMul(MatC,MatD);
785  printf("MatC * MatD\n");
786  MatPrint(MatR);
787  MatFree(MatC);
788  MatFree(MatD);
789  MatFree(MatR);
790  sep();
791  printf("\n");
792 
793 // ===============================================================
794 // ===============================================================
795 // Test functions required for 3 point screen calibration
796 // See: https://www.maximintegrated.com/en/app-notes/index.mvp/id/5296
797 // Calibration in touch-screen systems
798  sep();
799  printf("Set of three display positions\n");
800  MatX = MatLoad(X3,3,1);
801  printf("X\n");
802  MatPrint(MatX);
803 
804  MatY = MatLoad(Y3,3,1);
805  printf("Y\n");
806  MatPrint(MatY);
807 
808  printf("Correponding touch screen positions, differing scale and skew\n");
809  MatA = MatLoadSQ(A3,3);
810  printf("A\n");
811  MatPrint(MatA);
812 
813 // 1/A
814  MatI = Invert(MatA);
815  printf("1/(A)\n");
816  MatPrint(MatI);
817  MatFree(MatA);
818 
819  printf("Solution Matrix to translate touch screen to screen positions\n");
820 // MatR = 1/A * X
821  MatR = MatMul(MatI,MatX);
822  printf("1/A * X\n");
823  MatPrint(MatR);
824  MatFree(MatR);
825  MatFree(MatX);
826 
827 // MatR = 1/A * Y
828  MatR = MatMul(MatI,MatY);
829  printf("1/A * Y\n");
830  MatPrint(MatR);
831  MatFree(MatR);
832  MatFree(MatY);
833 
834  MatFree(MatI);
835  sep();
836  printf("\n");
837 
838 // ===============================================================
839 // ===============================================================
840 // Test functions required for N point screen calibration
841 // Use least square solution by Pseudo Invert method
842 // See: https://www.maximintegrated.com/en/app-notes/index.mvp/id/5296
843 // Calibration in touch-screen systems
844  sep();
845  printf("Set of five display positions\n");
846  MatX = MatLoad(X5,5,1);
847  printf("X\n");
848  MatPrint(MatX);
849 
850  MatY = MatLoad(Y5,5,1);
851  printf("Y\n");
852  MatPrint(MatY);
853 
854  printf("Correponding touch screen positions, differing scale and skew\n");
855  MatA = MatLoad(A5,5,3);
856  printf("A\n");
857  MatPrint(MatA);
858 
859  printf("Compute pseudo-inverse matrix, 1/(AT � A) � AT\n");
860  MatPI = PseudoInvert(MatA);
861  printf("PI = Pseudo Invert(A)\n");
862  MatPrint(MatPI);
863  MatFree(MatA);
864 
865  printf("Solution Matrix to translate touch screen to screen positions\n");
866 // MatR = PI * X
867  MatR = MatMul(MatPI,MatX);
868  printf("(R = PI * X\n");
869  MatPrint(MatR);
870  MatFree(MatR);
871  MatFree(MatX);
872 
873 // MatR = 1/A * Y
874  MatR = MatMul(MatPI,MatY);
875  printf("R = PI * Y\n");
876  MatPrint(MatR);
877  MatFree(MatR);
878  MatFree(MatY);
879 
880  MatFree(MatPI);
881 
882  sep();
883  printf("\n");
884 
885 }
886 #endif
fopen
MEMSPACE FILE * fopen(const char *path, const char *mode)
POSIX Open a file with path name and ascii file mode string.
Definition: posix.c:801
MatLoadSQ
MEMSPACE mat_t MatLoadSQ(void *V, int size)
Load a square matrix.
Definition: matrix.c:179
printf
MEMSPACE int printf(const char *format,...)
_mat::data
float ** data
Definition: matrix.h:29
Transpose
MEMSPACE mat_t Transpose(mat_t MatA)
Transpose matrix.
Definition: matrix.c:268
strtod
MEMSPACE double strtod(const char *nptr, char **endptr)
Determinant
MEMSPACE float Determinant(mat_t MatA)
Determinant by recursion using Cofactors.
Definition: matrix.c:358
MEMSPACE
#define MEMSPACE
Definition: user_config.h:17
Invert
MEMSPACE mat_t Invert(mat_t MatA)
Calculate Matrix Inverse.
Definition: matrix.c:408
safecalloc
void * safecalloc(int size, int elements)
Safe Alloc - Display Error message if Calloc fails.
Definition: ram.c:122
fgets
MEMSPACE char * fgets(char *str, int size, FILE *stream)
get a string from stdin See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:432
fprintf
MEMSPACE int fprintf(FILE *fp, const char *format,...)
fprintf function Example user defined printf function using fputc for I/O This method allows I/O to d...
Definition: posix.c:2533
MatPrint
MEMSPACE void MatPrint(mat_t matrix)
Print a matrix.
Definition: matrix.c:190
DeleteRowCol
MEMSPACE mat_t DeleteRowCol(mat_t MatA, int row, int col)
Create smaller matrix by deleatting specified row and colume Used by Minor and Cofactor.
Definition: matrix.c:216
safefree
void safefree(void *p)
Safe free - Only free a pointer if it is in malloc memory range.
Definition: ram.c:158
MatWrite
MEMSPACE int MatWrite(char *name, mat_t MatW)
Write a matrix.
Definition: matrix.c:617
Adjugate
MEMSPACE mat_t Adjugate(mat_t MatA)
Adjugate is transpose of cofactor matrix of A.
Definition: matrix.c:333
_mat::rows
int rows
Definition: matrix.h:31
_mat
Definition: matrix.h:27
matrix.h
matrix functions
sum
MEMSPACE uint16_t sum(char *name)
MatMul
MEMSPACE mat_t MatMul(mat_t MatA, mat_t MatB)
Multiply two matrix.
Definition: matrix.c:500
NULL
#define NULL
Definition: user_config.h:85
MatLoad
MEMSPACE mat_t MatLoad(void *V, int rows, int cols)
Load a matrix.
Definition: matrix.c:154
MatAllocSQ
MEMSPACE mat_t MatAllocSQ(int size)
Allocate a matrix.
Definition: matrix.c:106
Cofactor
MEMSPACE float Cofactor(mat_t MatA, int row, int col)
Cofactor is determinate of minor submatrix * (-1)exp(row+col) Minor submatrix has one less row and co...
Definition: matrix.c:317
__file
FILE type structure.
Definition: posix.h:158
sep
MEMSPACE void sep()
print seperator
Definition: parsing.c:35
fclose
MEMSPACE int fclose(FILE *stream)
POSIX close a file stream.
Definition: posix.c:1261
lines
int lines
Config file line number.
Definition: drives.c:1368
main
int main(void)
main() for gpib project
Definition: main.c:507
Minor
MEMSPACE float Minor(mat_t MatA, int row, int col)
Compute determinate of the minor submatrix Minor submatrix has one less row and column as a result.
Definition: matrix.c:297
TestSquare
MEMSPACE int TestSquare(mat_t MatA)
Credits: https://www.cs.rochester.edu/~brown/Crypto/assts/projects/adj.html.
Definition: matrix.c:36
MatRead
MEMSPACE mat_t MatRead(char *name)
Read a matrix.
Definition: matrix.c:538
MatAlloc
MEMSPACE mat_t MatAlloc(int rows, int cols)
Allocate a matrix.
Definition: matrix.c:48
sscanf
int sscanf(const char *strp, const char *fmt,...)
_mat::size
int size
Definition: matrix.h:32
_mat::cols
int cols
Definition: matrix.h:30
MatFree
MEMSPACE void MatFree(mat_t matF)
Free a matrix.
Definition: matrix.c:117
PseudoInvert
MEMSPACE mat_t PseudoInvert(mat_t MatA)
Calculate Pseudo Matrix Inverse Used for least square fitting of non square matrix with excess soluti...
Definition: matrix.c:466