/***************************************************************************** FILE : $Source: /projects/higgs1/SNNS/CVS/SNNS/tools/sources/ff_bignet.c,v $ SHORTNAME : ff_bignet SNNS VERSION : 4.2 PURPOSE : Creates feedforward network NOTES : AUTHOR : Michael Vogt (xgui Version by Michael Schmalzl) DATE : 4.7.1997 CHANGED BY : Guenter Mamier RCS VERSION : $Revision: 1.11 $ LAST CHANGE : $Date: 1998/04/22 16:48:06 $ Copyright (c) 1990-1995 SNNS Group, IPVR, Univ. Stuttgart, FRG Copyright (c) 1996-1998 SNNS Group, WSI, Univ. Tuebingen, FRG ******************************************************************************/ #include #include #include #include #include #include "kr_typ.h" #include "kernel.h" #include "glob_typ.h" #include "kr_ui.h" #include "ff_bignet.ph" /***************************************************************************** FUNCTION : create_net_PROC PURPOSE : creates a SNNS-net in two steps: 1. calls up a function, which creates SNNS-units 2. calls up a function, which connections the SNNS-units. NOTES : There are nine different ways (PLANE_TO_PLANE, PLANE_TO_CLUSTER ...) two different planes can be connected. RETURNS : UPDATE : 20.12.1991 ******************************************************************************/ static void create_net_PROC(void) { LINK *LINK_element = LINK_first_element; PLANE *source_plane, *target_plane; int move,type_of_connection=1; if(PLANE_length == 0) return; calculate_first_snns_unit_no_of_plane(); create_snns_unit(); while(LINK_element != NULL) { move = (*LINK_element).move; type_of_connection = (*LINK_element).type_of_connection; source_plane = get_plane((*LINK_element).SOURCE.plane); target_plane = get_plane((*LINK_element).TARGET.plane); switch(type_of_connection) { case PLANE_TO_PLANE: make_link(source_plane, 0, 0, (*source_plane).width, (*source_plane).height, target_plane, 0, 0, (*target_plane).width,(*target_plane).height); break; case PLANE_TO_CLUSTER: make_link(source_plane, 0, 0, (*source_plane).width, (*source_plane).height, target_plane, (*LINK_element).TARGET.CLUSTER.x-1, (*LINK_element).TARGET.CLUSTER.y-1, (*LINK_element).TARGET.CLUSTER.width, (*LINK_element).TARGET.CLUSTER.height); break; case PLANE_TO_UNIT: make_link(source_plane, 0, 0, (*source_plane).width, (*source_plane).height, target_plane, (*LINK_element).TARGET.UNIT.x-1, (*LINK_element).TARGET.UNIT.y-1, 1, 1); break; case CLUSTER_TO_PLANE: make_link(source_plane, (*LINK_element).SOURCE.CLUSTER.x-1, (*LINK_element).SOURCE.CLUSTER.y-1, (*LINK_element).SOURCE.CLUSTER.width, (*LINK_element).SOURCE.CLUSTER.height, target_plane, 0, 0, (*target_plane).width, (*target_plane).height); break; case CLUSTER_TO_CLUSTER: if(move) { make_move_link(source_plane, (*LINK_element).SOURCE.CLUSTER.x-1, (*LINK_element).SOURCE.CLUSTER.y-1, (*LINK_element).SOURCE.CLUSTER.width, (*LINK_element).SOURCE.CLUSTER.height, target_plane, (*LINK_element).TARGET.CLUSTER.x-1, (*LINK_element).TARGET.CLUSTER.y-1, (*LINK_element).TARGET.CLUSTER.width, (*LINK_element).TARGET.CLUSTER.height, (*LINK_element).SOURCE.MOVE.delta_x, (*LINK_element).SOURCE.MOVE.delta_y, (*LINK_element).TARGET.MOVE.delta_x, (*LINK_element).TARGET.MOVE.delta_y); } else { make_link(source_plane, (*LINK_element).SOURCE.CLUSTER.x-1, (*LINK_element).SOURCE.CLUSTER.y-1, (*LINK_element).SOURCE.CLUSTER.width, (*LINK_element).SOURCE.CLUSTER.height, target_plane, (*LINK_element).TARGET.CLUSTER.x-1, (*LINK_element).TARGET.CLUSTER.y-1, (*LINK_element).TARGET.CLUSTER.width, (*LINK_element).TARGET.CLUSTER.height); } break; case CLUSTER_TO_UNIT: if(move) { make_move_link(source_plane, (*LINK_element).SOURCE.CLUSTER.x-1, (*LINK_element).SOURCE.CLUSTER.y-1, (*LINK_element).SOURCE.CLUSTER.width, (*LINK_element).SOURCE.CLUSTER.height, target_plane, (*LINK_element).TARGET.UNIT.x-1, (*LINK_element).TARGET.UNIT.y-1, 1, 1, (*LINK_element).SOURCE.MOVE.delta_x, (*LINK_element).SOURCE.MOVE.delta_y, (*LINK_element).TARGET.MOVE.delta_x, (*LINK_element).TARGET.MOVE.delta_y); } else { make_link(source_plane, (*LINK_element).SOURCE.CLUSTER.x-1, (*LINK_element).SOURCE.CLUSTER.y-1, (*LINK_element).SOURCE.CLUSTER.width, (*LINK_element).SOURCE.CLUSTER.height, target_plane,(*LINK_element).TARGET.UNIT.x-1, (*LINK_element).TARGET.UNIT.y-1, 1, 1); } break; case UNIT_TO_PLANE: make_link(source_plane, (*LINK_element).SOURCE.UNIT.x-1, (*LINK_element).SOURCE.UNIT.y-1, 1, 1, target_plane, 0, 0, (*target_plane).width, (*target_plane).height); break; case UNIT_TO_CLUSTER: if(move) { make_move_link(source_plane, (*LINK_element).SOURCE.UNIT.x-1, (*LINK_element).SOURCE.UNIT.y-1, 1, 1, target_plane, (*LINK_element).TARGET.CLUSTER.x-1, (*LINK_element).TARGET.CLUSTER.y-1, (*LINK_element).TARGET.CLUSTER.width, (*LINK_element).TARGET.CLUSTER.height, (*LINK_element).SOURCE.MOVE.delta_x, (*LINK_element).SOURCE.MOVE.delta_y, (*LINK_element).TARGET.MOVE.delta_x, (*LINK_element).TARGET.MOVE.delta_y); } else { make_link(source_plane, (*LINK_element).SOURCE.UNIT.x-1, (*LINK_element).SOURCE.UNIT.y-1, 1, 1, target_plane, (*LINK_element).TARGET.CLUSTER.x-1, (*LINK_element).TARGET.CLUSTER.y-1, (*LINK_element).TARGET.CLUSTER.width, (*LINK_element).TARGET.CLUSTER.height); } break; case UNIT_TO_UNIT: if(move) { make_move_link(source_plane, (*LINK_element).SOURCE.UNIT.x-1, (*LINK_element).SOURCE.UNIT.y-1, 1, 1, target_plane, (*LINK_element).TARGET.UNIT.x-1, (*LINK_element).TARGET.UNIT.y-1, 1, 1, (*LINK_element).SOURCE.MOVE.delta_x, (*LINK_element).SOURCE.MOVE.delta_y, (*LINK_element).TARGET.MOVE.delta_x, (*LINK_element).TARGET.MOVE.delta_y); } else { make_link(source_plane, (*LINK_element).SOURCE.UNIT.x-1, (*LINK_element).SOURCE.UNIT.y-1, 1, 1, target_plane,(*LINK_element).TARGET.UNIT.x-1, (*LINK_element).TARGET.UNIT.y-1, 1, 1); } break; } /* switch */ LINK_element = (*LINK_element).next; } /* while */ } /***************************************************************************** FUNCTION : calculate_first_snns_unit_no_of_plane PURPOSE : calculates the SNNS unit no of the left upper corner of every plane. NOTES : relative to this unit no, knowing the width and the height of the planes, all other SNNS unit numbers of every plane can be calculated. That's important for the function create_snns_unit. RETURNS : UPDATE : 20.12.1991 ******************************************************************************/ static void calculate_first_snns_unit_no_of_plane(void) { PLANE *PLANE_element = PLANE_first_element; (*PLANE_element).begin = 1; PLANE_element = (*PLANE_element).next; while(PLANE_element != NULL) { (*PLANE_element).begin = (*(*PLANE_element).before).width * (*(*PLANE_element).before).height + (*(*PLANE_element).before).begin; PLANE_element = (*PLANE_element).next; } } /***************************************************************************** FUNCTION : calculate_x_begin PURPOSE : calculates the x-coordinate of a plane, where the plane is positioned in the display window. plane. NOTES : RETURNS : The X-Position UPDATE : 20.12.1991 ******************************************************************************/ static int calculate_x_begin(int *new_x_begin, int *old_x_begin, int *x_max, int width, int pos) { if(pos == BN_RIGHT) { *new_x_begin = *x_max + 2; *x_max = *new_x_begin + width; *old_x_begin = *new_x_begin; return *new_x_begin; } else if(pos == BN_BELOW) { if(*x_max < (*old_x_begin + width)) { *x_max = *old_x_begin + width; } return *old_x_begin; } else if(pos == BN_LEFT) { *x_max = 0; *old_x_begin = 2; *new_x_begin = *x_max + 2; *x_max = *new_x_begin + width; return *new_x_begin; } return 0; } /***************************************************************************** FUNCTION : calculate_y_begin PURPOSE : calculates the y-coordinate of a plane, where the plane is positioned in the display window. plane. NOTES : RETURNS : The Y-Position UPDATE : 20.12.1991 ******************************************************************************/ static int calculate_y_begin(int *absolute_y_max, int *relative_y_max, int *y_offset, int height, int pos) { int y_begin; if(pos == BN_RIGHT) { y_begin = *y_offset + 2; *relative_y_max = y_begin + height; if(*absolute_y_max < *relative_y_max) { *absolute_y_max = *relative_y_max; } } else if(pos == BN_BELOW) { y_begin = *relative_y_max + 2; *relative_y_max = y_begin + height; if(*absolute_y_max < *relative_y_max) { *absolute_y_max = *relative_y_max; } } else if(pos == BN_LEFT) { *y_offset = *absolute_y_max; y_begin = *y_offset + 2; *absolute_y_max = y_begin + height; *relative_y_max = y_begin + height; } return y_begin; } /***************************************************************************** FUNCTION : create_snns_unit PURPOSE : creates for every unit of a plane a SNNS default unit. NOTES : RETURNS : UPDATE : 20.12.1991 ******************************************************************************/ static void create_snns_unit(void) { struct PosType unit_pos; PLANE *PLANE_element = PLANE_first_element; int x,y,width,height,unit_no,pos,ret; int y_offset=0,relative_y_max=0,absolute_y_max=0,y_begin; int x_max=0,old_x_begin=2,new_x_begin,x_begin; ret = krui_allocateUnits((*PLANE_last_element).width * (*PLANE_last_element).height + (*PLANE_last_element).begin-1); if(ret != 0){ fprintf(stderr, "%s\n", krui_error(ret)); exit(1); } while(PLANE_element != NULL) { height = (*PLANE_element).height; width = (*PLANE_element).width; pos = (*PLANE_element).pos; y_begin = calculate_y_begin(&absolute_y_max, &relative_y_max, &y_offset, height, pos); x_begin = calculate_x_begin(&new_x_begin, &old_x_begin, &x_max, width, pos); for(y=0;y 0 && ((*x)+width+x_step<=(*plane_no).width)) (*x)+= x_step; else if(y_step > 0 && ((*y)+height+y_step<=(*plane_no).height)){ (*y) += y_step; (*x) = x_pivot; }else return 0; return 1; } /***************************************************************************** FUNCTION : more_data PURPOSE : read command line integer value or "+" NOTES : RETURNS : UPDATE : ******************************************************************************/ static int more_data(char **rest, char **argv[], int *argc) { int retval = 0; if (*argc > 0 && strcmp(**argv, "+") != 0) { retval = atoi(**argv); *rest = **argv; while (**rest != '\0' && isspace((int) **rest)) (*rest)++; while (**rest != '\0' && isdigit((int) **rest)) (*rest)++; (*argc)--; (*argv)++; } return retval; } /***************************************************************************** FUNCTION : read_link_elements PURPOSE : reads the input datas of the link editor (Edit Link). NOTES : RETURNS : @@@ UPDATE : 20.12.1991 ******************************************************************************/ #define RETURN_IF_FAULT(x) { if ((x) != NULL && *(x) != '\0') return 0; } static int read_link_elements(int *argc, char **argv[]) { /* the + character may be used to fill the remaining parameters of one link definition column (compare bignet for xgui). e.g. argv values like '1 + 2 +' result in a full connection between plane 1 and plane 2 */ char *rest = NULL; if (*argc <= 0) return 0; LINK_source_plane = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_source_cluster_x = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_source_cluster_y = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_source_cluster_width = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_source_cluster_height = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_source_unit_x = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_source_unit_y = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_source_move_delta_x = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_source_move_delta_y = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); if (*argc > 0 && strcmp(**argv, "+") == 0) { (*argc)--; (*argv)++; } if (*argc <= 0) return 0; LINK_target_plane = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_target_cluster_x = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_target_cluster_y = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_target_cluster_width = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_target_cluster_height = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_target_unit_x = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_target_unit_y = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_target_move_delta_x = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); LINK_target_move_delta_y = more_data(&rest, argv, argc); RETURN_IF_FAULT(rest); if (*argc > 0 && strcmp(**argv, "+") == 0) { (*argc)--; (*argv)++; } return correct_link_input(); } #undef RETURN_IF_FAULT /***************************************************************************** FUNCNCTION : correct_link_input PURPOSE : tests whether the input datas of Edit Link are correct and decides what kind of link (PLANE_TO_PLANE, PLANE_TO_LINK...) we have. NOTES : RETURNS : Correct = 1 ; ERROR = 0 UPDATE : 20.12.1991 ******************************************************************************/ static int correct_link_input(void) { PLANE *PLANE_source_no, *PLANE_target_no; int link_type; int target_plane=0, source_plane=0; int target_cluster=0, source_cluster=0; int target_unit=0, source_unit=0; int a,b,c,d; PLANE_source_no = get_plane(LINK_source_plane); PLANE_target_no = get_plane(LINK_target_plane); LINK_move = 0; /* check plane numbers */ if((PLANE_source_no == NULL) || (PLANE_target_no == NULL)) return 0; /* check source plane dimensions */ if((LINK_source_cluster_x + LINK_source_cluster_width > (*PLANE_source_no).width + 1) || (LINK_source_cluster_y + LINK_source_cluster_height > (*PLANE_source_no).height + 1) || (LINK_source_unit_x > (*PLANE_source_no).width) || (LINK_source_unit_y > (*PLANE_source_no).height)) return 0; /* check target plane dimensions */ if((LINK_target_cluster_x + LINK_target_cluster_width > (*PLANE_target_no).width + 1) || (LINK_target_cluster_y + LINK_target_cluster_height > (*PLANE_target_no).height + 1) || (LINK_target_unit_x > (*PLANE_target_no).width) || (LINK_target_unit_y > (*PLANE_target_no).height)) return 0; target_plane = TARGET_PLANE; source_plane = SOURCE_PLANE; a = b = c = d = 0; if(((a=LINK_source_cluster_x) >= 1) && ((b=LINK_source_cluster_y) >= 1) && ((c=LINK_source_cluster_width) >= 1) && ((d=LINK_source_cluster_height) >= 1)) { source_cluster = SOURCE_CLUSTER; } if((! source_cluster) && a+b+c+d) return 0; a = b = 0; if(((a=LINK_source_unit_x) >= 1) && ((b=LINK_source_unit_y) >= 1)) source_unit = SOURCE_UNIT; if((! source_unit) && a+b) return 0; a = b = c = d = 0; if(((a=LINK_target_cluster_x) >= 1) && ((b=LINK_target_cluster_y) >= 1) && ((c=LINK_target_cluster_width) >= 1) && ((d=LINK_target_cluster_height) >= 1)) { target_cluster = TARGET_CLUSTER; } if((! target_cluster) && a+b+c+d) return 0; a = b = 0; if(((a=LINK_target_unit_x) >= 1) && ((b=LINK_target_unit_y) >= 1)) target_unit = TARGET_UNIT; if((! target_unit) && a+b) return 0; link_type = target_plane + source_plane + target_cluster+ source_cluster + target_unit + source_unit; if((link_type == PLANE_TO_PLANE) || (link_type == PLANE_TO_CLUSTER) || (link_type == PLANE_TO_UNIT) || (link_type == CLUSTER_TO_PLANE) || (link_type == CLUSTER_TO_CLUSTER) || (link_type == CLUSTER_TO_UNIT) || (link_type == UNIT_TO_PLANE) ||(link_type == UNIT_TO_CLUSTER) || (link_type == UNIT_TO_UNIT)) { LINK_type_of_connection = link_type; if((LINK_source_move_delta_x || LINK_source_move_delta_y || LINK_target_move_delta_x || LINK_target_move_delta_y) && ((link_type==CLUSTER_TO_CLUSTER) || (link_type==CLUSTER_TO_UNIT) || (link_type==UNIT_TO_CLUSTER) || (link_type == UNIT_TO_UNIT))) { LINK_move = 1; } else { LINK_source_move_delta_x = LINK_source_move_delta_y = LINK_target_move_delta_x = LINK_target_move_delta_y = 0; } } return 1; } /***************************************************************************** FUNCTION : enter_link PURPOSE : enters a link element at the end of the link list. NOTES : RETURNS : UPDATE : 20.12.1991 ******************************************************************************/ static void enter_link(void) { LINK *LINK_element; LINK_element = (LINK *)malloc(sizeof(LINK)); (*LINK_element).SOURCE.plane = LINK_source_plane; (*LINK_element).SOURCE.CLUSTER.x = LINK_source_cluster_x; (*LINK_element).SOURCE.CLUSTER.y = LINK_source_cluster_y; (*LINK_element).SOURCE.CLUSTER.width = LINK_source_cluster_width; (*LINK_element).SOURCE.CLUSTER.height = LINK_source_cluster_height; (*LINK_element).SOURCE.UNIT.x = LINK_source_unit_x; (*LINK_element).SOURCE.UNIT.y = LINK_source_unit_y; (*LINK_element).SOURCE.MOVE.delta_x = LINK_source_move_delta_x; (*LINK_element).SOURCE.MOVE.delta_y = LINK_source_move_delta_y; (*LINK_element).TARGET.plane = LINK_target_plane; (*LINK_element).TARGET.CLUSTER.x = LINK_target_cluster_x; (*LINK_element).TARGET.CLUSTER.y = LINK_target_cluster_y; (*LINK_element).TARGET.CLUSTER.width = LINK_target_cluster_width; (*LINK_element).TARGET.CLUSTER.height = LINK_target_cluster_height; (*LINK_element).TARGET.UNIT.x = LINK_target_unit_x; (*LINK_element).TARGET.UNIT.y = LINK_target_unit_y; (*LINK_element).TARGET.MOVE.delta_x = LINK_target_move_delta_x; (*LINK_element).TARGET.MOVE.delta_y = LINK_target_move_delta_y; (*LINK_element).move = LINK_move; (*LINK_element).type_of_connection = LINK_type_of_connection; (*LINK_element).next = NULL; (*LINK_element).before = LINK_last_element; if(LINK_last_element != NULL) { (*LINK_last_element).next = LINK_element; } if(LINK_first_element == NULL) { LINK_first_element = LINK_element; } LINK_last_element = LINK_element; LINK_current_element = LINK_last_element; LINK_length++; }/* enter_link */ /***************************************************************************** FUNCTION : read_plane_elements PURPOSE : reads the input datas of the plane editor (Edit Plane). NOTES : RETURNS : Error = 0 ; Succes = 1 UPDATE : 20.12.1991 ******************************************************************************/ static int read_plane_elements(int *argc, char **argv[]) { if (*argc > 0) { PLANE_width = atoi(**argv); (*argv)++; (*argc)--; } else return 0; if (*argc > 0) { PLANE_height = atoi(**argv); (*argv)++; (*argc)--; } else return 0; PLANE_type = BN_HIDDEN; PLANE_type_was_set = 0; PLANE_act_func = DEFAULT_ACT_FUNC; PLANE_out_func = DEFAULT_OUT_FUNC; if (*argc > 0 && krui_isFunction(**argv, ACT_FUNC)) { PLANE_act_func = **argv; (*argv)++; (*argc)--; if (*argc > 0 && krui_isFunction(**argv, OUT_FUNC)) { PLANE_out_func = **argv; (*argv)++; (*argc)--; if (*argc > 0) { if (strcasecmp(**argv, "input") == 0) { PLANE_type = BN_INPUT; PLANE_type_was_set = 1; (*argv)++; (*argc)--; } else if (strcasecmp(**argv, "hidden") == 0) { PLANE_type = BN_HIDDEN; PLANE_type_was_set = 1; (*argv)++; (*argc)--; } else if (strcasecmp(**argv, "output") == 0) { PLANE_type = BN_OUTPUT; PLANE_type_was_set = 1; (*argv)++; (*argc)--; } } } } return (PLANE_width > 0 && PLANE_height > 0) ? 1 : 0; } /***************************************************************************** FUNCTION : enter_plane PURPOSE : enters a plane element at the end of the plane list. NOTES : RETURNS : UPDATE : 20.12.1991 ******************************************************************************/ static void enter_plane(void) { PLANE *PLANE_element; PLANE_element = (PLANE *)malloc(sizeof(PLANE)); (*PLANE_element).width = PLANE_width; (*PLANE_element).height = PLANE_height; (*PLANE_element).pos = BN_RIGHT; (*PLANE_element).type = PLANE_type; (*PLANE_element).type_was_set = PLANE_type_was_set; (*PLANE_element).act_func = PLANE_act_func; (*PLANE_element).out_func = PLANE_out_func; (*PLANE_element).next = NULL; (*PLANE_element).before = PLANE_last_element; if(PLANE_last_element != NULL) { (*PLANE_last_element).next = PLANE_element; } if(PLANE_first_element == NULL) { PLANE_first_element = PLANE_element; } PLANE_last_element = PLANE_element; PLANE_current_element = PLANE_last_element; PLANE_no = ++PLANE_length; } /***************************************************************************** FUNCTION : get_plane PURPOSE : returns a pointer to the plane element "plane_no" NOTES : RETURNS : returns a pointer to the plane element "plane_no" UPDATE : 20.12.1991 ******************************************************************************/ static PLANE *get_plane(int plane_no) { PLANE *PLANE_element = PLANE_first_element; int counter = 1; while((PLANE_element != NULL) && (counter != plane_no)) { counter++; PLANE_element = (*PLANE_element).next; } return PLANE_element; } /***************************************************************************** FUNCTION : usage PURPOSE : gives usage information NOTES : RETURNS : UPDATE : ******************************************************************************/ void usage(void) { fprintf(stderr, "usage: ff_bignet ... ... []\n"); fprintf(stderr, "\n"); fprintf(stderr, " : -p [ [ []]]\n"); fprintf(stderr, " : number of units in x-direction\n"); fprintf(stderr, " : number of units in y-direction\n"); fprintf(stderr, " : opt. activation function\n"); fprintf(stderr, " e.g.: Act_Logistic\n"); fprintf(stderr, " : opt. activation function, must be given too\n"); fprintf(stderr, " e.g.: Out_Identity\n"); fprintf(stderr, " : opt. layer type, and must be given too\n"); fprintf(stderr, " valid types: input, hidden, or output\n\n"); fprintf(stderr, " : -l ... [+] ... [+]\n"); fprintf(stderr, " Link parameters must be given in the following order.\n"); fprintf(stderr, " All remaing Parameters within one section may be filled\n"); fprintf(stderr, " with 0 by giving the '+' character (0 means: not used)\n\n"); fprintf(stderr, " Source section:\n"); fprintf(stderr, " : source plane (1, 2, ...)\n"); fprintf(stderr, " : x position of source cluster\n"); fprintf(stderr, " : y position of source cluster\n"); fprintf(stderr, " : width of source cluster\n"); fprintf(stderr, " : height of source cluster\n"); fprintf(stderr, " : x position of a distinct source unit\n"); fprintf(stderr, " : y position of a distinct source unit\n"); fprintf(stderr, " : delta x for multiple source fields\n"); fprintf(stderr, " : delta y for multiple source fields\n\n"); fprintf(stderr, " Target section:\n"); fprintf(stderr, " : target plane (1, 2, ...)\n"); fprintf(stderr, " : x position of target cluster\n"); fprintf(stderr, " : y position of target cluster\n"); fprintf(stderr, " : width of target cluster\n"); fprintf(stderr, " : height of target cluster\n"); fprintf(stderr, " : x position of a distinct target unit\n"); fprintf(stderr, " : y position of a distinct target unit\n"); fprintf(stderr, " : delta x for multiple target fields\n"); fprintf(stderr, " : delta y for multiple target fields\n\n"); fprintf(stderr, " : name of the output file (default SNNS_FF_NET.net)\n"); exit(1); } /***************************************************************************** FUNCTION : main PURPOSE : NOTES : RETURNS : UPDATE : ******************************************************************************/ void main(int argc, char *argv[]) { char *file_name; int err; file_name = "SNNS_FF_NET.net"; if (argc <= 1) usage(); argc--; argv++; while (argc > 0) { if (strcmp(*argv, "-p") == 0) { argc--; argv++; if (!read_plane_elements(&argc, &argv)){ fprintf(stderr, "Illegal plane definition\n"); exit(1); } enter_plane(); } else if (strcmp(*argv, "-l") == 0) { argc--; argv++; if (!read_link_elements(&argc, &argv)){ fprintf(stderr, "Illegal link definition\n"); exit(1); } enter_link(); } else { if (argc != 1) { fprintf(stderr, "Parameters not understood. Invalid function or type name?\n"); exit(1); } file_name = *argv; argc--; argv++; } } if(PLANE_first_element != NULL && !PLANE_first_element->type_was_set) { PLANE_first_element -> type = BN_INPUT; } if(PLANE_last_element != NULL && !PLANE_last_element->type_was_set) { PLANE_last_element -> type = BN_OUTPUT; } create_net_PROC(); err = krui_setUpdateFunc("Topological_Order"); if (err != KRERR_NO_ERROR){ fprintf(stderr, "%s\n", krui_error(err)); exit(1); } err = krui_setLearnFunc("Std_Backpropagation"); if (err != KRERR_NO_ERROR){ fprintf(stderr, "%s\n", krui_error(err)); exit(1); } err = krui_saveNet(file_name, "SNNS_FF_NET"); if (err != KRERR_NO_ERROR){ fprintf(stderr, "%s\n", krui_error(err)); exit(1); } exit(0); } /* end of file */