sigset.c 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360
  1. #include <string.h>
  2. #include <stdlib.h>
  3. #include <grass/imagery.h>
  4. #include <grass/gis.h>
  5. static int gettag(FILE *, char *);
  6. static int get_nbands(FILE *, struct SigSet *);
  7. static int get_title(FILE *, struct SigSet *);
  8. static int get_class(FILE *, struct SigSet *);
  9. static int get_classnum(FILE *, struct ClassSig *);
  10. static int get_classtype(FILE *, struct ClassSig *);
  11. static int get_classtitle(FILE *, struct ClassSig *);
  12. static int get_subclass(FILE *, struct SigSet *, struct ClassSig *);
  13. static int get_subclass_pi(FILE *, struct SubSig *);
  14. static int get_subclass_means(FILE *, struct SubSig *, int);
  15. static int get_subclass_covar(FILE *, struct SubSig *, int);
  16. static double **alloc_matrix(int rows, int cols)
  17. {
  18. double **m;
  19. int i;
  20. m = (double **)G_calloc(rows, sizeof(double *));
  21. m[0] = (double *)G_calloc(rows * cols, sizeof(double));
  22. for (i = 1; i < rows; i++)
  23. m[i] = m[i - 1] + cols;
  24. return m;
  25. }
  26. int I_SigSetNClasses(struct SigSet *S)
  27. {
  28. int i, count;
  29. for (i = 0, count = 0; i < S->nclasses; i++)
  30. if (S->ClassSig[i].used)
  31. count++;
  32. return count;
  33. }
  34. struct ClassData *I_AllocClassData(struct SigSet *S,
  35. struct ClassSig *C, int npixels)
  36. {
  37. struct ClassData *Data;
  38. Data = &(C->ClassData);
  39. Data->npixels = npixels;
  40. Data->count = 0;
  41. Data->x = alloc_matrix(npixels, S->nbands);
  42. Data->p = alloc_matrix(npixels, C->nsubclasses);
  43. return Data;
  44. }
  45. int I_InitSigSet(struct SigSet *S)
  46. {
  47. S->nbands = 0;
  48. S->nclasses = 0;
  49. S->ClassSig = NULL;
  50. S->title = NULL;
  51. return 0;
  52. }
  53. int I_SigSetNBands(struct SigSet *S, int nbands)
  54. {
  55. S->nbands = nbands;
  56. return 0;
  57. }
  58. struct ClassSig *I_NewClassSig(struct SigSet *S)
  59. {
  60. struct ClassSig *Sp;
  61. if (S->nclasses == 0)
  62. S->ClassSig = (struct ClassSig *)G_malloc(sizeof(struct ClassSig));
  63. else
  64. S->ClassSig = (struct ClassSig *)G_realloc((char *)S->ClassSig,
  65. sizeof(struct ClassSig) *
  66. (S->nclasses + 1));
  67. Sp = &S->ClassSig[S->nclasses++];
  68. Sp->classnum = 0;
  69. Sp->nsubclasses = 0;
  70. Sp->used = 1;
  71. Sp->type = SIGNATURE_TYPE_MIXED;
  72. Sp->title = NULL;
  73. return Sp;
  74. }
  75. struct SubSig *I_NewSubSig(struct SigSet *S, struct ClassSig *C)
  76. {
  77. struct SubSig *Sp;
  78. int i;
  79. if (C->nsubclasses == 0)
  80. C->SubSig = (struct SubSig *)G_malloc(sizeof(struct SubSig));
  81. else
  82. C->SubSig = (struct SubSig *)G_realloc((char *)C->SubSig,
  83. sizeof(struct SubSig) *
  84. (C->nsubclasses + 1));
  85. Sp = &C->SubSig[C->nsubclasses++];
  86. Sp->used = 1;
  87. Sp->R = (double **)G_calloc(S->nbands, sizeof(double *));
  88. Sp->R[0] = (double *)G_calloc(S->nbands * S->nbands, sizeof(double));
  89. for (i = 1; i < S->nbands; i++)
  90. Sp->R[i] = Sp->R[i - 1] + S->nbands;
  91. Sp->Rinv = (double **)G_calloc(S->nbands, sizeof(double *));
  92. Sp->Rinv[0] = (double *)G_calloc(S->nbands * S->nbands, sizeof(double));
  93. for (i = 1; i < S->nbands; i++)
  94. Sp->Rinv[i] = Sp->Rinv[i - 1] + S->nbands;
  95. Sp->means = (double *)G_calloc(S->nbands, sizeof(double));
  96. Sp->N = 0;
  97. Sp->pi = 0;
  98. Sp->cnst = 0;
  99. return Sp;
  100. }
  101. #define eq(a,b) strcmp(a,b)==0
  102. int I_ReadSigSet(FILE * fd, struct SigSet *S)
  103. {
  104. char tag[256];
  105. I_InitSigSet(S);
  106. while (gettag(fd, tag)) {
  107. if (eq(tag, "title:"))
  108. if (get_title(fd, S) != 0)
  109. return -1;
  110. if (eq(tag, "nbands:"))
  111. if (get_nbands(fd, S) != 0)
  112. return -1;
  113. if (eq(tag, "class:"))
  114. if (get_class(fd, S) != 0)
  115. return -1;
  116. }
  117. return 1; /* for now assume success */
  118. }
  119. static int gettag(FILE * fd, char *tag)
  120. {
  121. if (fscanf(fd, "%s", tag) != 1)
  122. return 0;
  123. G_strip(tag);
  124. return 1;
  125. }
  126. static int get_nbands(FILE * fd, struct SigSet *S)
  127. {
  128. if (fscanf(fd, "%d", &S->nbands) != 1)
  129. return -1;
  130. return 0;
  131. }
  132. static int get_title(FILE * fd, struct SigSet *S)
  133. {
  134. char title[1024];
  135. *title = 0;
  136. if (fscanf(fd, "%[^\n]", title) != 1)
  137. return -1;
  138. I_SetSigTitle(S, title);
  139. return 0;
  140. }
  141. static int get_class(FILE * fd, struct SigSet *S)
  142. {
  143. char tag[1024];
  144. struct ClassSig *C;
  145. C = I_NewClassSig(S);
  146. while (gettag(fd, tag)) {
  147. if (eq(tag, "endclass:"))
  148. break;
  149. if (eq(tag, "classnum:"))
  150. if (get_classnum(fd, C) != 0)
  151. return -1;
  152. if (eq(tag, "classtype:"))
  153. if (get_classtype(fd, C) != 0)
  154. return -1;
  155. if (eq(tag, "classtitle:"))
  156. if (get_classtitle(fd, C) != 0)
  157. return -1;
  158. if (eq(tag, "subclass:"))
  159. if (get_subclass(fd, S, C) != 0)
  160. return -1;
  161. }
  162. return 0;
  163. }
  164. static int get_classnum(FILE * fd, struct ClassSig *C)
  165. {
  166. if (fscanf(fd, "%ld", &C->classnum) != 1)
  167. return -1;
  168. return 0;
  169. }
  170. static int get_classtype(FILE * fd, struct ClassSig *C)
  171. {
  172. if (fscanf(fd, "%d", &C->type) != 1)
  173. return -1;
  174. return 0;
  175. }
  176. static int get_classtitle(FILE * fd, struct ClassSig *C)
  177. {
  178. char title[1024];
  179. *title = 0;
  180. if (fscanf(fd, "%[^\n]", title) != 1)
  181. return -1;
  182. I_SetClassTitle(C, title);
  183. return 0;
  184. }
  185. static int get_subclass(FILE * fd, struct SigSet *S, struct ClassSig *C)
  186. {
  187. struct SubSig *Sp;
  188. char tag[1024];
  189. Sp = I_NewSubSig(S, C);
  190. while (gettag(fd, tag)) {
  191. if (eq(tag, "endsubclass:"))
  192. break;
  193. if (eq(tag, "pi:"))
  194. if (get_subclass_pi(fd, Sp) != 0)
  195. return -1;
  196. if (eq(tag, "means:"))
  197. if (get_subclass_means(fd, Sp, S->nbands) != 0)
  198. return -1;
  199. if (eq(tag, "covar:"))
  200. if (get_subclass_covar(fd, Sp, S->nbands) != 0)
  201. return -1;
  202. }
  203. return 0;
  204. }
  205. static int get_subclass_pi(FILE * fd, struct SubSig *Sp)
  206. {
  207. if (fscanf(fd, "%lf", &Sp->pi) != 1)
  208. return -1;
  209. return 0;
  210. }
  211. static int get_subclass_means(FILE * fd, struct SubSig *Sp, int nbands)
  212. {
  213. int i;
  214. for (i = 0; i < nbands; i++)
  215. if (fscanf(fd, "%lf", &Sp->means[i]) != 1)
  216. return -1;
  217. return 0;
  218. }
  219. static int get_subclass_covar(FILE * fd, struct SubSig *Sp, int nbands)
  220. {
  221. int i, j;
  222. for (i = 0; i < nbands; i++)
  223. for (j = 0; j < nbands; j++)
  224. if (fscanf(fd, "%lf", &Sp->R[i][j]) != 1)
  225. return -1;
  226. return 0;
  227. }
  228. int I_SetSigTitle(struct SigSet *S, const char *title)
  229. {
  230. if (title == NULL)
  231. title = "";
  232. if (S->title)
  233. free(S->title);
  234. S->title = G_store(title);
  235. return 0;
  236. }
  237. const char *I_GetSigTitle(const struct SigSet *S)
  238. {
  239. if (S->title)
  240. return S->title;
  241. else
  242. return "";
  243. }
  244. int I_SetClassTitle(struct ClassSig *C, const char *title)
  245. {
  246. if (title == NULL)
  247. title = "";
  248. if (C->title)
  249. free(C->title);
  250. C->title = G_store(title);
  251. return 0;
  252. }
  253. const char *I_GetClassTitle(const struct ClassSig *C)
  254. {
  255. if (C->title)
  256. return C->title;
  257. else
  258. return "";
  259. }
  260. int I_WriteSigSet(FILE * fd, const struct SigSet *S)
  261. {
  262. const struct ClassSig *Cp;
  263. const struct SubSig *Sp;
  264. int i, j, b1, b2;
  265. fprintf(fd, "title: %s\n", I_GetSigTitle(S));
  266. fprintf(fd, "nbands: %d\n", S->nbands);
  267. for (i = 0; i < S->nclasses; i++) {
  268. Cp = &S->ClassSig[i];
  269. if (!Cp->used)
  270. continue;
  271. if (Cp->nsubclasses <= 0)
  272. continue;
  273. fprintf(fd, "class:\n");
  274. fprintf(fd, " classnum: %ld\n", Cp->classnum);
  275. fprintf(fd, " classtitle: %s\n", I_GetClassTitle(Cp));
  276. fprintf(fd, " classtype: %d\n", Cp->type);
  277. for (j = 0; j < Cp->nsubclasses; j++) {
  278. Sp = &Cp->SubSig[j];
  279. fprintf(fd, " subclass:\n");
  280. fprintf(fd, " pi: %g\n", Sp->pi);
  281. fprintf(fd, " means:");
  282. for (b1 = 0; b1 < S->nbands; b1++)
  283. fprintf(fd, " %g", Sp->means[b1]);
  284. fprintf(fd, "\n");
  285. fprintf(fd, " covar:\n");
  286. for (b1 = 0; b1 < S->nbands; b1++) {
  287. fprintf(fd, " ");
  288. for (b2 = 0; b2 < S->nbands; b2++)
  289. fprintf(fd, " %g", Sp->R[b1][b2]);
  290. fprintf(fd, "\n");
  291. }
  292. fprintf(fd, " endsubclass:\n");
  293. }
  294. fprintf(fd, "endclass:\n");
  295. }
  296. return 0;
  297. }