1 /* 2 * Copyright 2002, Sebastian Nozzi <sebnozzi@gmx.net>. 3 * Copyright 2004, Francois Revol. 4 * 5 * Distributed under the terms of the MIT License. 6 */ 7 8 #include <stdio.h> 9 #include <string.h> 10 11 #include <FindDirectory.h> 12 #include <fs_info.h> 13 14 15 #define NO_ERRORS 0 16 #define ARGUMENT_MISSING 1 17 #define WRONG_DIR_TYPE 2 18 19 typedef struct { 20 const char *key; 21 directory_which value; 22 } directoryType; 23 24 #define KEYVALUE_PAIR(key) {#key, key} 25 26 directoryType directoryTypes[] = { 27 // Generic directories 28 KEYVALUE_PAIR(B_DESKTOP_DIRECTORY), 29 KEYVALUE_PAIR(B_TRASH_DIRECTORY), 30 KEYVALUE_PAIR(B_APPS_DIRECTORY), 31 KEYVALUE_PAIR(B_PREFERENCES_DIRECTORY), 32 KEYVALUE_PAIR(B_UTILITIES_DIRECTORY), 33 34 // System directories 35 KEYVALUE_PAIR(B_SYSTEM_DIRECTORY), 36 KEYVALUE_PAIR(B_SYSTEM_ADDONS_DIRECTORY), 37 KEYVALUE_PAIR(B_SYSTEM_BOOT_DIRECTORY), 38 KEYVALUE_PAIR(B_SYSTEM_FONTS_DIRECTORY), 39 KEYVALUE_PAIR(B_SYSTEM_LIB_DIRECTORY), 40 KEYVALUE_PAIR(B_SYSTEM_SERVERS_DIRECTORY), 41 KEYVALUE_PAIR(B_SYSTEM_APPS_DIRECTORY), 42 KEYVALUE_PAIR(B_SYSTEM_BIN_DIRECTORY), 43 KEYVALUE_PAIR(B_SYSTEM_DOCUMENTATION_DIRECTORY), 44 KEYVALUE_PAIR(B_SYSTEM_PREFERENCES_DIRECTORY), 45 KEYVALUE_PAIR(B_SYSTEM_TRANSLATORS_DIRECTORY), 46 KEYVALUE_PAIR(B_SYSTEM_MEDIA_NODES_DIRECTORY), 47 KEYVALUE_PAIR(B_SYSTEM_SOUNDS_DIRECTORY), 48 KEYVALUE_PAIR(B_SYSTEM_DATA_DIRECTORY), 49 50 // Common directories 51 KEYVALUE_PAIR(B_COMMON_DIRECTORY), 52 KEYVALUE_PAIR(B_COMMON_SYSTEM_DIRECTORY), 53 KEYVALUE_PAIR(B_COMMON_ADDONS_DIRECTORY), 54 KEYVALUE_PAIR(B_COMMON_BOOT_DIRECTORY), 55 KEYVALUE_PAIR(B_COMMON_FONTS_DIRECTORY), 56 KEYVALUE_PAIR(B_COMMON_LIB_DIRECTORY), 57 KEYVALUE_PAIR(B_COMMON_SERVERS_DIRECTORY), 58 KEYVALUE_PAIR(B_COMMON_BIN_DIRECTORY), 59 KEYVALUE_PAIR(B_COMMON_ETC_DIRECTORY), 60 KEYVALUE_PAIR(B_COMMON_DOCUMENTATION_DIRECTORY), 61 KEYVALUE_PAIR(B_COMMON_SETTINGS_DIRECTORY), 62 KEYVALUE_PAIR(B_COMMON_DEVELOP_DIRECTORY), 63 KEYVALUE_PAIR(B_COMMON_LOG_DIRECTORY), 64 KEYVALUE_PAIR(B_COMMON_SPOOL_DIRECTORY), 65 KEYVALUE_PAIR(B_COMMON_TEMP_DIRECTORY), 66 KEYVALUE_PAIR(B_COMMON_VAR_DIRECTORY), 67 KEYVALUE_PAIR(B_COMMON_TRANSLATORS_DIRECTORY), 68 KEYVALUE_PAIR(B_COMMON_MEDIA_NODES_DIRECTORY), 69 KEYVALUE_PAIR(B_COMMON_SOUNDS_DIRECTORY), 70 KEYVALUE_PAIR(B_COMMON_DATA_DIRECTORY), 71 72 // User directories 73 KEYVALUE_PAIR(B_USER_DIRECTORY), 74 KEYVALUE_PAIR(B_USER_CONFIG_DIRECTORY), 75 KEYVALUE_PAIR(B_USER_ADDONS_DIRECTORY), 76 KEYVALUE_PAIR(B_USER_BOOT_DIRECTORY), 77 KEYVALUE_PAIR(B_USER_FONTS_DIRECTORY), 78 KEYVALUE_PAIR(B_USER_LIB_DIRECTORY), 79 KEYVALUE_PAIR(B_USER_SETTINGS_DIRECTORY), 80 KEYVALUE_PAIR(B_USER_DESKBAR_DIRECTORY), 81 KEYVALUE_PAIR(B_USER_PRINTERS_DIRECTORY), 82 KEYVALUE_PAIR(B_USER_TRANSLATORS_DIRECTORY), 83 KEYVALUE_PAIR(B_USER_MEDIA_NODES_DIRECTORY), 84 KEYVALUE_PAIR(B_USER_SOUNDS_DIRECTORY), 85 KEYVALUE_PAIR(B_USER_DATA_DIRECTORY), 86 KEYVALUE_PAIR(B_USER_CACHE_DIRECTORY), 87 88 // Legacy system directories 89 KEYVALUE_PAIR(B_BEOS_DIRECTORY), 90 KEYVALUE_PAIR(B_BEOS_SYSTEM_DIRECTORY), 91 KEYVALUE_PAIR(B_BEOS_ADDONS_DIRECTORY), 92 KEYVALUE_PAIR(B_BEOS_BOOT_DIRECTORY), 93 KEYVALUE_PAIR(B_BEOS_FONTS_DIRECTORY), 94 KEYVALUE_PAIR(B_BEOS_LIB_DIRECTORY), 95 KEYVALUE_PAIR(B_BEOS_SERVERS_DIRECTORY), 96 KEYVALUE_PAIR(B_BEOS_APPS_DIRECTORY), 97 KEYVALUE_PAIR(B_BEOS_BIN_DIRECTORY), 98 KEYVALUE_PAIR(B_BEOS_ETC_DIRECTORY), 99 KEYVALUE_PAIR(B_BEOS_DOCUMENTATION_DIRECTORY), 100 KEYVALUE_PAIR(B_BEOS_PREFERENCES_DIRECTORY), 101 KEYVALUE_PAIR(B_BEOS_TRANSLATORS_DIRECTORY), 102 KEYVALUE_PAIR(B_BEOS_MEDIA_NODES_DIRECTORY), 103 KEYVALUE_PAIR(B_BEOS_SOUNDS_DIRECTORY), 104 KEYVALUE_PAIR(B_BEOS_DATA_DIRECTORY), 105 106 {NULL, B_USER_DESKBAR_DIRECTORY} 107 }; 108 109 110 static void 111 listDirectoryWhich(void) 112 { 113 int i; 114 115 for (i = 0; directoryTypes[i].key; i++) { 116 printf("%s\n", directoryTypes[i].key); 117 } 118 } 119 120 121 static bool 122 retrieveDirValue(directoryType *list, const char *key, 123 directory_which *valueOut) 124 { 125 unsigned i = 0; 126 127 while (list[i].key != NULL) { 128 if (strcmp(list[i].key, key) == 0) { 129 *valueOut = list[i].value; 130 return true; 131 } 132 133 i++; 134 } 135 136 return false; 137 } 138 139 140 static void 141 usageMsg() 142 { 143 printf("usage: /bin/finddir -l | [ -v volume ] directory_which\n"); 144 printf("\t-l\t list valid which constants to use\n"); 145 printf("\t-v <file> use the specified volume for directory\n"); 146 printf("\t\t constants that are volume-specific.\n"); 147 printf("\t\t <file> can be any file on that volume.\n"); 148 printf("\t\t defaults to the boot volume.\n"); 149 printf(" For a description of recognized directory_which constants,\n"); 150 printf(" see the find_directory(...) documentation in the Be Book.\n"); 151 } 152 153 154 int 155 main(int argc, char *argv[]) 156 { 157 int directoryArgNr; 158 int status; 159 dev_t volume; 160 directory_which dirType; 161 int returnCode; 162 163 status = NO_ERRORS; 164 directoryArgNr = 1; 165 returnCode = 0; 166 167 dirType = B_BEOS_DIRECTORY; /* so that it compiles */ 168 169 /* By default use boot volume*/ 170 volume = dev_for_path("/boot"); 171 172 if (argc <= 1) { 173 status = ARGUMENT_MISSING; 174 } else { 175 if (strcmp(argv[1], "-l") == 0 ) { 176 listDirectoryWhich(); 177 return 0; 178 } 179 if (strcmp(argv[1], "-v") == 0 ) { 180 if (argc >= 3) { 181 dev_t temp_volume; 182 /* get volume from second arg */ 183 temp_volume = dev_for_path(argv[2]); 184 185 /* Keep default value in case of error */ 186 if (temp_volume >= 0) 187 volume = temp_volume; 188 189 /* two arguments were used for volume */ 190 directoryArgNr+=2; 191 } else { 192 /* set status to argument missing */ 193 status = ARGUMENT_MISSING; 194 } 195 } 196 } 197 198 if (status == NO_ERRORS && argc > directoryArgNr) { 199 /* get directory constant from next argument */ 200 201 if (!retrieveDirValue(directoryTypes, argv[directoryArgNr], &dirType)) 202 status = WRONG_DIR_TYPE; 203 } else { 204 status = ARGUMENT_MISSING; 205 } 206 207 /* Do the actual directoy finding */ 208 209 if (status == NO_ERRORS) { 210 /* Question: would B_PATH_NAME_LENGTH alone have been enough? */ 211 char buffer[B_PATH_NAME_LENGTH+B_FILE_NAME_LENGTH]; 212 status_t result = find_directory (dirType, volume, false, buffer, 213 sizeof(buffer)); 214 if (result == B_OK) { 215 printf("%s\n", buffer); 216 } else { 217 /* else what? */ 218 /* this can not happen! */ 219 fprintf(stderr, "Serious internal error; contact support\n"); 220 } 221 } 222 223 /* Error messages and return code setting */ 224 225 if (status == WRONG_DIR_TYPE) { 226 fprintf(stderr, "%s: unrecognized directory_which constant \'%s\'\n", argv[0], 227 argv[directoryArgNr]); 228 returnCode = 252; 229 } 230 231 if (status == ARGUMENT_MISSING) { 232 usageMsg(); 233 returnCode = 255; 234 } 235 236 return returnCode; 237 } 238 239