Skip to content
Snippets Groups Projects
say.c 186 KiB
Newer Older
  • Learn to ignore specific revisions
  • 			else
    				snprintf(fn, sizeof(fn), "digits/1000000S");
     
    			if ((num % 1000000) &&
    
    				((!((num / 1000) % 1000) && ((num % 1000) < 100 || !(num % 100))) ||
    
    				(!(num % 1000) && (((num / 1000) % 1000) < 100 || !((num / 1000) % 100))) ) )
    				playh = 1;
    			num = num % 1000000;
    		}
    
    		if (!res) {
    			if (!ast_streamfile(chan, fn, language)) {
    
    				if ((audiofd > -1) && (ctrlfd > -1))
    
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);	
    				else
    					res = ast_waitstream(chan, ints);
    			}
    			ast_stopstream(chan);
    		}
    
    		if (!res && playh) {
    			res = wait_file(chan, ints, "digits/pt-e", language);
    			ast_stopstream(chan);
    			playh = 0;
    		}
    
    /*! \brief  ast_say_number_full_se: Swedish syntax */
    
    static int ast_say_number_full_se(struct ast_channel *chan, int num, const char *ints, const char *language, const char *options, int audiofd, int ctrlfd)
    
    {
    	int res = 0;
    	int playh = 0;
    	char fn[256] = "";
    
    	int cn = 1;		/* +1 = commune; -1 = neuter */
    
    	if (!num) 
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    	if (options && !strncasecmp(options, "n",1)) cn = -1;
    
    	while(!res && (num || playh)) {
    
    		if (num < 0) {
    			snprintf(fn, sizeof(fn), "digits/minus");
    			if ( num > INT_MIN ) {
    				num = -num;
    			} else {
    
    			}	
    		} else if (playh) {
    			snprintf(fn, sizeof(fn), "digits/hundred");
    			playh = 0;
    		} else if (num < 20) {
    			snprintf(fn, sizeof(fn), "digits/%d", num);
    			num = 0;
    		} else if (num < 100) {
    			snprintf(fn, sizeof(fn), "digits/%d", (num /10) * 10);
    			num -= ((num / 10) * 10);
    		} else if (num == 1 && cn == -1) {	/* En eller ett? */
    		 	snprintf(fn, sizeof(fn), "digits/1N");
    			num = 0;
    		} else {
    			if (num < 1000){
    				snprintf(fn, sizeof(fn), "digits/%d", (num/100));
    				playh++;
    				num -= ((num / 100) * 100);
    
    				if (num < 1000000) { /* 1,000,000 */
    					res = ast_say_number_full_se(chan, num / 1000, ints, language, options, audiofd, ctrlfd);
    					if (res) {
    						return res;
    					}
    					num = num % 1000;
    					snprintf(fn, sizeof(fn), "digits/thousand");
    
    					if (num < 1000000000) {	/* 1,000,000,000 */
    						res = ast_say_number_full_se(chan, num / 1000000, ints, language, options, audiofd, ctrlfd);
    						if (res) {
    
    						num = num % 1000000;
    						snprintf(fn, sizeof(fn), "digits/million");
    					} else {
    						ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    						res = -1;
    
    		}
    		if (!res) {
    			if(!ast_streamfile(chan, fn, language)) {
    
    				if ((audiofd > -1) && (ctrlfd > -1))
    
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    				else
    					res = ast_waitstream(chan, ints);
    				ast_stopstream(chan);
    			}
    		}
    
    /*! \brief  ast_say_number_full_tw: Taiwanese / Chinese syntax */
    
    static int ast_say_number_full_tw(struct ast_channel *chan, int num, const char *ints, const char *language, int audiofd, int ctrlfd)
    
    {
    	int res = 0;
    	int playh = 0;
    	char fn[256] = "";
    	if (!num)
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    
    	while(!res && (num || playh)) {
    
    			if (num < 0) {
    				snprintf(fn, sizeof(fn), "digits/minus");
    				if ( num > INT_MIN ) {
    					num = -num;
    				} else {
    					num = 0;
    				}	
    			} else if (playh) {
    
    				snprintf(fn, sizeof(fn), "digits/hundred");
    				playh = 0;
    			} else	if (num < 10) {
    				snprintf(fn, sizeof(fn), "digits/%d", num);
    				num = 0;
    			} else	if (num < 100) {
    				snprintf(fn, sizeof(fn), "digits/%d", (num /10) * 10);
    				num -= ((num / 10) * 10);
    			} else {
    				if (num < 1000){
    					snprintf(fn, sizeof(fn), "digits/%d", (num/100));
    					playh++;
    					num -= ((num / 100) * 100);
    				} else {
    					if (num < 1000000) { /* 1,000,000 */
    
    						res = ast_say_number_full_tw(chan, num / 1000, ints, language, audiofd, ctrlfd);
    
    						if (res)
    							return res;
    						num = num % 1000;
    						snprintf(fn, sizeof(fn), "digits/thousand");
    					} else {
    						if (num < 1000000000) {	/* 1,000,000,000 */
    
    							res = ast_say_number_full_tw(chan, num / 1000000, ints, language, audiofd, ctrlfd);
    
    							if (res)
    								return res;
    							num = num % 1000000;
    							snprintf(fn, sizeof(fn), "digits/million");
    						} else {
    							ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    							res = -1;
    						}
    					}
    				}
    			}
    			if (!res) {
    				if(!ast_streamfile(chan, fn, language)) {
    
    					if ((audiofd > -1) && (ctrlfd > -1))
    
    						res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    					else
    						res = ast_waitstream(chan, ints);
    				}
    				ast_stopstream(chan);
    			}
    	}
    	return res;
    }
    
    
    /*! \brief  determine last digits for thousands/millions (ru) */
    
    static int get_lastdigits_ru(int num) {
    	if (num < 20) {
    		return num;
    	} else if (num < 100) {
    		return get_lastdigits_ru(num % 10);
    	} else if (num < 1000) {
    		return get_lastdigits_ru(num % 100);
    	}
    	return 0;	/* number too big */
    }
    
    
    
    /*! \brief  ast_say_number_full_ru: Russian syntax */
    /*! \brief  additional files:
    
    	n00.gsm			(one hundred, two hundred, ...)
    	thousand.gsm
    	million.gsm
    	thousands-i.gsm		(tisyachi)
    	million-a.gsm		(milliona)
    	thousands.gsm
    	millions.gsm
    	1f.gsm			(odna)
    	2f.gsm			(dve)
        
    	where 'n' from 1 to 9
    */
    static int ast_say_number_full_ru(struct ast_channel *chan, int num, const char *ints, const char *language, const char *options, int audiofd, int ctrlfd)
    {
    	int res = 0;
    	int lastdigits = 0;
    	char fn[256] = "";
    	if (!num) 
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    
    	while(!res && (num)) {
    		if (num < 0) {
    			snprintf(fn, sizeof(fn), "digits/minus");
    			if ( num > INT_MIN ) {
    				num = -num;
    			} else {
    				num = 0;
    			}	
    		} else	if (num < 20) {
    			if(options && strlen(options) == 1 && num < 3) {
    			    snprintf(fn, sizeof(fn), "digits/%d%s", num, options);
    			} else {
        			    snprintf(fn, sizeof(fn), "digits/%d", num);
    			}
    			num = 0;
    		} else	if (num < 100) {
    			snprintf(fn, sizeof(fn), "digits/%d", num - (num % 10));
    			num %= 10;
    		} else 	if (num < 1000){
    			snprintf(fn, sizeof(fn), "digits/%d", num - (num % 100));
    			num %= 100;
    		} else 	if (num < 1000000) { /* 1,000,000 */
    			lastdigits = get_lastdigits_ru(num / 1000);
    			/* say thousands */
    			if (lastdigits < 3) {
    				res = ast_say_number_full_ru(chan, num / 1000, ints, language, "f", audiofd, ctrlfd);
    			} else {
    				res = ast_say_number_full_ru(chan, num / 1000, ints, language, NULL, audiofd, ctrlfd);
    			}
    			if (res)
    				return res;
    			if (lastdigits == 1) {
    				snprintf(fn, sizeof(fn), "digits/thousand");
    			} else if (lastdigits > 1 && lastdigits < 5) {
    				snprintf(fn, sizeof(fn), "digits/thousands-i");
    			} else {
    				snprintf(fn, sizeof(fn), "digits/thousands");
    			}
    			num %= 1000;
    		} else 	if (num < 1000000000) {	/* 1,000,000,000 */
    			lastdigits = get_lastdigits_ru(num / 1000000);
    			/* say millions */
    			res = ast_say_number_full_ru(chan, num / 1000000, ints, language, NULL, audiofd, ctrlfd);
    			if (res)
    				return res;
    			if (lastdigits == 1) {
    				snprintf(fn, sizeof(fn), "digits/million");
    			} else if (lastdigits > 1 && lastdigits < 5) {
    				snprintf(fn, sizeof(fn), "digits/million-a");
    			} else {
    				snprintf(fn, sizeof(fn), "digits/millions");
    			}
    			num %= 1000000;
    		} else {
    			ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    				res = -1;
    		}
    		if (!res) {
    			if (!ast_streamfile(chan, fn, language)) {
    				if ((audiofd  > -1) && (ctrlfd > -1))
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    				else
    					res = ast_waitstream(chan, ints);
    			}
    			ast_stopstream(chan);
    		}
    	}
    	return res;
    }
    
    
    
    /*! \brief  ast_say_enumeration_full: call language-specific functions */
    
    /* Called from AGI */
    int ast_say_enumeration_full(struct ast_channel *chan, int num, const char *ints, const char *language, const char *options, int audiofd, int ctrlfd)
    {
    	if (!strcasecmp(language,"en") ) {	/* English syntax */
    	   return(ast_say_enumeration_full_en(chan, num, ints, language, audiofd, ctrlfd));
    
    	} else if (!strcasecmp(language, "da") ) {	/* Danish syntax */
    	   return(ast_say_enumeration_full_da(chan, num, ints, language, options, audiofd, ctrlfd));
    
    	} else if (!strcasecmp(language, "de") ) {	/* German syntax */
    	   return(ast_say_enumeration_full_de(chan, num, ints, language, options, audiofd, ctrlfd));
    	} 
    	
    	/* Default to english */
    	return(ast_say_enumeration_full_en(chan, num, ints, language, audiofd, ctrlfd));
    }
    
    /*! \brief  ast_say_enumeration: call language-specific functions without file descriptors */
    
    int ast_say_enumeration(struct ast_channel *chan, int num, const char *ints, const char *language, const char *options)
    
    	return(ast_say_enumeration_full(chan, num, ints, language, options, -1, -1));
    }
    
    
    /*! \brief  ast_say_enumeration_full_en: English syntax */
    
    /* This is the default syntax, if no other syntax defined in this file is used */
    static int ast_say_enumeration_full_en(struct ast_channel *chan, int num, const char *ints, const char *language, int audiofd, int ctrlfd)
    {
    	int res = 0, t = 0;
    
    	char fn[256] = "";
    	
    
    	while(!res && num) {
    		if (num < 0) {
    			snprintf(fn, sizeof(fn), "digits/minus"); /* kind of senseless for enumerations, but our best effort for error checking */
    			if ( num > INT_MIN ) {
    				num = -num;
    			} else {
    				num = 0;
    			}	
    
    		} else if (num < 20) {
    
    			snprintf(fn, sizeof(fn), "digits/h-%d", num);
    
    		} else if (num < 100) { 
    			int tens = num / 10;
    			num = num % 10;
    			if (num == 0) {
    				snprintf(fn, sizeof(fn), "digits/h-%d", (tens * 10));
    			} else {
    				snprintf(fn, sizeof(fn), "digits/%d", (tens * 10));
    			}
    
    		} else if (num < 1000) {
    
    			int hundreds = num / 100;
    			num = num % 100;
    			if (hundreds > 1 || t == 1) {
    				res = ast_say_number_full_en(chan, hundreds, ints, language, audiofd, ctrlfd);
    			}			
    			if (res)
    				return res;
    			if (num) {
    				snprintf(fn, sizeof(fn), "digits/hundred");
    
    				snprintf(fn, sizeof(fn), "digits/h-hundred");
    
    		} else if (num < 1000000) {
    			int thousands = num / 1000;
    			num = num % 1000;
    			if (thousands > 1 || t == 1) {
    				res = ast_say_number_full_en(chan, thousands, ints, language, audiofd, ctrlfd);
    
    			if (res)
    				return res;
    			if (num) {					
    				snprintf(fn, sizeof(fn), "digits/thousand");
    			} else {
    				snprintf(fn, sizeof(fn), "digits/h-thousand");
    
    			t = 1;
    		} else if (num < 1000000000) {
    			int millions = num / 1000000;
    			num = num % 1000000;
    			t = 1;
    			res = ast_say_number_full_en(chan, millions, ints, language, audiofd, ctrlfd);
    			if (res)
    				return res;
    			if (num) {					
    				snprintf(fn, sizeof(fn), "digits/million");
    			} else {
    				snprintf(fn, sizeof(fn), "digits/h-million");
    
    		} else if (num < INT_MAX) {
    			int billions = num / 1000000000;
    			num = num % 1000000000;
    			t = 1;
    			res = ast_say_number_full_en(chan, billions, ints, language, audiofd, ctrlfd);
    			if (res)
    				return res;
    			if (num) {					
    				snprintf(fn, sizeof(fn), "digits/billion");
    			} else {
    				snprintf(fn, sizeof(fn), "digits/h-billion");
    
    		} else if (num == INT_MAX) {
    			snprintf(fn, sizeof(fn), "digits/h-last");
    			num = 0;
    		} else {
    			ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    			res = -1;
    
    		if (!res) {
    
    			if (!ast_streamfile(chan, fn, language)) {
    
    				if ((audiofd > -1) && (ctrlfd > -1)) {
    
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    				} else {
    					res = ast_waitstream(chan, ints);
    				}
    			}
    			ast_stopstream(chan);
    
    /*! \brief  ast_say_enumeration_full_da: Danish syntax */
    
    static int ast_say_enumeration_full_da(struct ast_channel *chan, int num, const char *ints, const char *language, const char *options, int audiofd, int ctrlfd)
    {
    	/* options can be: '' or 'm' male gender; 'f' female gender; 'n' neuter gender; 'p' plural */
    	int res = 0, t = 0;
    	char fn[256] = "", fna[256] = "";
    	char *gender;
    
    	if (options && !strncasecmp(options, "f",1)) {
    		gender = "F";
    	} else if (options && !strncasecmp(options, "n",1)) {
    		gender = "N";
    	} else {
    		gender = "";
    	}
    
    	if (!num) 
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    
    	while(!res && num) {
    		if (num < 0) {
    			snprintf(fn, sizeof(fn), "digits/minus"); /* kind of senseless for enumerations, but our best effort for error checking */
    			if ( num > INT_MIN ) {
    				num = -num;
    			} else {
    				num = 0;
    			}	
    		} else if (num < 100 && t) {
    			snprintf(fn, sizeof(fn), "digits/and");
    			t = 0;
    		} else if (num < 20) {
    			snprintf(fn, sizeof(fn), "digits/h-%d%s", num, gender);
    			num = 0;
    		} else if (num < 100) {
    			int ones = num % 10;
    			if (ones) {
    				snprintf(fn, sizeof(fn), "digits/%d-and", ones);
    				num -= ones;
    			} else {
    				snprintf(fn, sizeof(fn), "digits/h-%d%s", num, gender);
    				num = 0;
    			}
    		} else if (num == 100 && t == 0) {
    			snprintf(fn, sizeof(fn), "digits/h-hundred%s", gender);
    			num = 0;
    		} else if (num < 1000) {
    			int hundreds = num / 100;
    			num = num % 100;
    			if (hundreds == 1) {
    				snprintf(fn, sizeof(fn), "digits/1N");
    			} else {
    				snprintf(fn, sizeof(fn), "digits/%d", hundreds);
    			}
    			if (num) {					
    				snprintf(fna, sizeof(fna), "digits/hundred");
    			} else {
    				snprintf(fna, sizeof(fna), "digits/h-hundred%s", gender);
    			}
    			t = 1;
    		} else 	if (num < 1000000) {
    			int thousands = num / 1000;
    			num = num % 1000;
    			if (thousands == 1) {
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/1N");
    					snprintf(fna, sizeof(fna), "digits/thousand");
    				} else {
    					if (t) {
    						snprintf(fn, sizeof(fn), "digits/1N");
    						snprintf(fna, sizeof(fna), "digits/h-thousand%s", gender);
    					} else {
    						snprintf(fn, sizeof(fn), "digits/h-thousand%s", gender);
    					}
    				}
    			} else {
    				res = ast_say_number_full_de(chan, thousands, ints, language, options, audiofd, ctrlfd);
    				if (res) {
    					return res;
    				}
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/thousand");
    				} else {
    					snprintf(fn, sizeof(fn), "digits/h-thousand%s", gender);
    				}
    			}
    			t = 1;
    		} else if (num < 1000000000) {
    			int millions = num / 1000000;
    			num = num % 1000000;
    			if (millions == 1) {
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/1F");
    					snprintf(fna, sizeof(fna), "digits/million");
    				} else {
    					snprintf(fn, sizeof(fn), "digits/1N");
    					snprintf(fna, sizeof(fna), "digits/h-million%s", gender);
    				}
    			} else {
    				res = ast_say_number_full_de(chan, millions, ints, language, options, audiofd, ctrlfd);
    				if (res) {
    					return res;
    				}
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/millions");
    				} else {
    					snprintf(fn, sizeof(fn), "digits/h-million%s", gender);
    				}
    			}
    			t = 1;
    		} else if (num < INT_MAX) {
    			int billions = num / 1000000000;
    			num = num % 1000000000;
    			if (billions == 1) {
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/1F");
    					snprintf(fna, sizeof(fna), "digits/milliard");
    				} else {
    					snprintf(fn, sizeof(fn), "digits/1N");
    					snprintf(fna, sizeof(fna), "digits/h-milliard%s", gender);
    				}
    			} else {
    				res = ast_say_number_full_de(chan, billions, ints, language, options, audiofd, ctrlfd);
    				if (res)
    					return res;
    				if (num) {					
    					snprintf(fn, sizeof(fna), "digits/milliards");
    				} else {
    					snprintf(fn, sizeof(fna), "digits/h-milliard%s", gender);
    				}
    			}
    			t = 1;
    		} else if (num == INT_MAX) {
    			snprintf(fn, sizeof(fn), "digits/h-last%s", gender);
    			num = 0;
    		} else {
    			ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    			res = -1;
    		}
    
    		if (!res) {
    			if (!ast_streamfile(chan, fn, language)) {
    				if ((audiofd > -1) && (ctrlfd > -1)) 
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    				else  
    					res = ast_waitstream(chan, ints);
    			}
    			ast_stopstream(chan);
    			if (!res) {
    				if (strlen(fna) != 0 && !ast_streamfile(chan, fna, language)) {
    					if ((audiofd > -1) && (ctrlfd > -1)) {
    						res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    					} else {
    						res = ast_waitstream(chan, ints);
    					}
    				}
    				ast_stopstream(chan);
    				strcpy(fna, "");
    			}
    		}
    	}
    	return res;
    }
    
    
    /*! \brief  ast_say_enumeration_full_de: German syntax */
    
    static int ast_say_enumeration_full_de(struct ast_channel *chan, int num, const char *ints, const char *language, const char *options, int audiofd, int ctrlfd)
    
    	/* options can be: '' or 'm' male gender; 'f' female gender; 'n' neuter gender; 'p' plural */
    	int res = 0, t = 0;
    	char fn[256] = "", fna[256] = "";
    	char *gender;
    
    	if (options && !strncasecmp(options, "f",1)) {
    		gender = "F";
    	} else if (options && !strncasecmp(options, "n",1)) {
    		gender = "N";
    	} else {
    		gender = "";
    
    	if (!num) 
    		return ast_say_digits_full(chan, 0,ints, language, audiofd, ctrlfd);
    
    Mark Spencer's avatar
    Mark Spencer committed
    
    
    	while(!res && num) {
    		if (num < 0) {
    			snprintf(fn, sizeof(fn), "digits/minus"); /* kind of senseless for enumerations, but our best effort for error checking */
    			if ( num > INT_MIN ) {
    				num = -num;
    			} else {
    				num = 0;
    			}	
    		} else if (num < 100 && t) {
    			snprintf(fn, sizeof(fn), "digits/and");
    			t = 0;
    		} else if (num < 20) {
    			snprintf(fn, sizeof(fn), "digits/h-%d%s", num, gender);
    			num = 0;
    		} else if (num < 100) {
    			int ones = num % 10;
    			if (ones) {
    				snprintf(fn, sizeof(fn), "digits/%d-and", ones);
    				num -= ones;
    			} else {
    				snprintf(fn, sizeof(fn), "digits/h-%d%s", num, gender);
    				num = 0;
    			}
    		} else if (num == 100 && t == 0) {
    			snprintf(fn, sizeof(fn), "digits/h-hundred%s", gender);
    			num = 0;
    		} else if (num < 1000) {
    			int hundreds = num / 100;
    			num = num % 100;
    			if (hundreds == 1) {
    				snprintf(fn, sizeof(fn), "digits/1N");
    			} else {
    				snprintf(fn, sizeof(fn), "digits/%d", hundreds);
    			}
    			if (num) {					
    				snprintf(fna, sizeof(fna), "digits/hundred");
    			} else {
    				snprintf(fna, sizeof(fna), "digits/h-hundred%s", gender);
    			}
    			t = 1;
    		} else 	if (num < 1000000) {
    			int thousands = num / 1000;
    			num = num % 1000;
    			if (thousands == 1) {
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/1N");
    					snprintf(fna, sizeof(fna), "digits/thousand");
    				} else {
    					if (t) {
    						snprintf(fn, sizeof(fn), "digits/1N");
    						snprintf(fna, sizeof(fna), "digits/h-thousand%s", gender);
    					} else {
    						snprintf(fn, sizeof(fn), "digits/h-thousand%s", gender);
    					}
    				}
    			} else {
    				res = ast_say_number_full_de(chan, thousands, ints, language, options, audiofd, ctrlfd);
    				if (res) {
    					return res;
    				}
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/thousand");
    				} else {
    					snprintf(fn, sizeof(fn), "digits/h-thousand%s", gender);
    				}
    			}
    			t = 1;
    		} else if (num < 1000000000) {
    			int millions = num / 1000000;
    			num = num % 1000000;
    			if (millions == 1) {
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/1F");
    					snprintf(fna, sizeof(fna), "digits/million");
    				} else {
    					snprintf(fn, sizeof(fn), "digits/1N");
    					snprintf(fna, sizeof(fna), "digits/h-million%s", gender);
    				}
    			} else {
    				res = ast_say_number_full_de(chan, millions, ints, language, options, audiofd, ctrlfd);
    				if (res) {
    					return res;
    				}
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/millions");
    				} else {
    					snprintf(fn, sizeof(fn), "digits/h-million%s", gender);
    				}
    			}
    			t = 1;
    		} else if (num < INT_MAX) {
    			int billions = num / 1000000000;
    			num = num % 1000000000;
    			if (billions == 1) {
    				if (num) {					
    					snprintf(fn, sizeof(fn), "digits/1F");
    					snprintf(fna, sizeof(fna), "digits/milliard");
    				} else {
    					snprintf(fn, sizeof(fn), "digits/1N");
    					snprintf(fna, sizeof(fna), "digits/h-milliard%s", gender);
    				}
    			} else {
    				res = ast_say_number_full_de(chan, billions, ints, language, options, audiofd, ctrlfd);
    				if (res)
    					return res;
    				if (num) {					
    					snprintf(fn, sizeof(fna), "digits/milliards");
    				} else {
    					snprintf(fn, sizeof(fna), "digits/h-milliard%s", gender);
    				}
    			}
    			t = 1;
    		} else if (num == INT_MAX) {
    			snprintf(fn, sizeof(fn), "digits/h-last%s", gender);
    			num = 0;
    		} else {
    			ast_log(LOG_DEBUG, "Number '%d' is too big for me\n", num);
    			res = -1;
    		}
    
    		if (!res) {
    			if (!ast_streamfile(chan, fn, language)) {
    
    				if ((audiofd > -1) && (ctrlfd > -1)) 
    
    					res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    				else  
    					res = ast_waitstream(chan, ints);
    			}
    			ast_stopstream(chan);
    			if (!res) {
    				if (strlen(fna) != 0 && !ast_streamfile(chan, fna, language)) {
    
    					if ((audiofd > -1) && (ctrlfd > -1)) {
    
    						res = ast_waitstream_full(chan, ints, audiofd, ctrlfd);
    					} else {
    						res = ast_waitstream(chan, ints);
    					}
    				}
    				ast_stopstream(chan);
    				strcpy(fna, "");
    			}
    		}
    	}
    	return res;
    }
    
    int ast_say_date(struct ast_channel *chan, time_t t, const char *ints, const char *lang)
    {
    	if (!strcasecmp(lang, "en") ) {	/* English syntax */
    		return(ast_say_date_en(chan, t, ints, lang));
    
    	} else if (!strcasecmp(lang, "da") ) {	/* Danish syntax */
    		return(ast_say_date_da(chan, t, ints, lang));
    
    	} else if (!strcasecmp(lang, "de") ) {	/* German syntax */
    		return(ast_say_date_de(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "fr") ) {	/* French syntax */
    		return(ast_say_date_fr(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "nl") ) {	/* Dutch syntax */
    		return(ast_say_date_nl(chan, t, ints, lang));
    	} else if (!strcasecmp(lang, "pt") ) {	/* Portuguese syntax */
    		return(ast_say_date_pt(chan, t, ints, lang));
    
    	} else if (!strcasecmp(lang, "gr") ) {  			/* Greek syntax */
    		return(ast_say_date_gr(chan, t, ints, lang));
    
    	}
    
    	/* Default to English */
    	return(ast_say_date_en(chan, t, ints, lang));
    }
    
    /* English syntax */
    int ast_say_date_en(struct ast_channel *chan, time_t t, const char *ints, const char *lang)
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	ast_localtime(&t,&tm,NULL);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    
    	if (!res)
    		res = ast_say_number(chan, tm.tm_mday, ints, lang, (char * ) NULL);
    
    	if (!res)
    		res = ast_waitstream(chan, ints);
    
    	if (!res)
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    	return res;
    }
    
    
    /* Danish syntax */
    int ast_say_date_da(struct ast_channel *chan, time_t t, const char *ints, const char *lang)
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	ast_localtime(&t,&tm,NULL);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res)
    		res = ast_say_enumeration(chan, tm.tm_mday, ints, lang, (char * ) NULL);
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res) {
    		/* Year */
    		int year = tm.tm_year + 1900;
    		if (year > 1999) {	/* year 2000 and later */
    			res = ast_say_number(chan, year, ints, lang, (char *) NULL);	
    		} else {
    			if (year < 1100) {
    				/* I'm not going to handle 1100 and prior */
    				/* We'll just be silent on the year, instead of bombing out. */
    			} else {
    			    /* year 1100 to 1999. will anybody need this?!? */
    				snprintf(fn,sizeof(fn), "digits/%d", (year / 100) );
    				res = wait_file(chan, ints, fn, lang);
    				if (!res) {
    					res = wait_file(chan,ints, "digits/hundred", lang);
    					if (!res && year % 100 != 0) {
    						res = ast_say_number(chan, (year % 100), ints, lang, (char *) NULL);	
    					}
    				}
    			}
    		}
    	}
    	return res;
    }
    
    
    /* German syntax */
    int ast_say_date_de(struct ast_channel *chan, time_t t, const char *ints, const char *lang)
    
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	ast_localtime(&t,&tm,NULL);
    
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    
    		res = ast_say_enumeration(chan, tm.tm_mday, ints, lang, (char * ) NULL);
    
    		res = ast_waitstream(chan, ints);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res) {
    		/* Year */
    		int year = tm.tm_year + 1900;
    		if (year > 1999) {	/* year 2000 and later */
    			res = ast_say_number(chan, year, ints, lang, (char *) NULL);	
    		} else {
    			if (year < 1100) {
    				/* I'm not going to handle 1100 and prior */
    				/* We'll just be silent on the year, instead of bombing out. */
    			} else {
    			    /* year 1100 to 1999. will anybody need this?!? */
    
    			    /* say 1967 as 'neunzehn hundert sieben und sechzig' */
    
    				snprintf(fn,sizeof(fn), "digits/%d", (year / 100) );
    				res = wait_file(chan, ints, fn, lang);
    				if (!res) {
    					res = wait_file(chan,ints, "digits/hundred", lang);
    					if (!res && year % 100 != 0) {
    						res = ast_say_number(chan, (year % 100), ints, lang, (char *) NULL);	
    					}
    				}
    			}
    		}
    	}
    	return res;
    }
    
    /* French syntax */
    int ast_say_date_fr(struct ast_channel *chan, time_t t, const char *ints, const char *lang)
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	ast_localtime(&t,&tm,NULL);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    
    		res = ast_say_number(chan, tm.tm_mday, ints, lang, (char * ) NULL);
    
    		res = ast_waitstream(chan, ints);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    
    	if (!res)
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    
    /* Dutch syntax */
    int ast_say_date_nl(struct ast_channel *chan, time_t t, const char *ints, const char *lang)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	ast_localtime(&t,&tm,NULL);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    	if (!res)
    		res = ast_say_number(chan, tm.tm_mday, ints, lang, (char * ) NULL);
    	if (!res) {
    		snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    		res = ast_streamfile(chan, fn, lang);
    		if (!res)
    			res = ast_waitstream(chan, ints);
    	}
    
    	if (!res)
    		res = ast_waitstream(chan, ints);
    	if (!res)
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    	return res;
    }
    
    /* Portuguese syntax */
    int ast_say_date_pt(struct ast_channel *chan, time_t t, const char *ints, const char *lang)
    {
    	struct tm tm;
    	char fn[256];
    	int res = 0;
    	ast_localtime(&t,&tm,NULL);
    	localtime_r(&t,&tm);
    	snprintf(fn, sizeof(fn), "digits/day-%d", tm.tm_wday);
    	if (!res)
    		res = wait_file(chan, ints, fn, lang);
    	if (!res)
    		res = ast_say_number(chan, tm.tm_mday, ints, lang, (char *) NULL);
    	if (!res)
    		res = wait_file(chan, ints, "digits/pt-de", lang);
    	snprintf(fn, sizeof(fn), "digits/mon-%d", tm.tm_mon);
    	if (!res)
    		res = wait_file(chan, ints, fn, lang);
    	if (!res)
    		res = wait_file(chan, ints, "digits/pt-de", lang);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	if (!res)
    		res = ast_say_number(chan, tm.tm_year + 1900, ints, lang, (char *) NULL);
    
    Mark Spencer's avatar
    Mark Spencer committed
    	return res;
    }
    
    
    int ast_say_date_with_format(struct ast_channel *chan, time_t time, const char *ints, const char *lang, const char *format, const char *timezone)
    
    	/* If no format is given, use default english format */
    	if (format == NULL)
    		format = "ABdY 'digits/at' IMp";
    
    
    	if (!strcasecmp(lang, "en") ) {	/* English syntax */
    		return(ast_say_date_with_format_en(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "da") ) {	/* Danish syntax */
    		return(ast_say_date_with_format_da(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "de") ) {	/* German syntax */
    		return(ast_say_date_with_format_de(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "es") || !strcasecmp(lang, "mx")) {	/* Spanish syntax */
    		return(ast_say_date_with_format_es(chan, time, ints, lang, format, timezone));
    
     	} else if (!strcasecmp(lang, "he")) {	/* Hebrew syntax */
     		return(ast_say_date_with_format_he(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "fr") ) {	/* French syntax */
    		return(ast_say_date_with_format_fr(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "it") ) {  /* Italian syntax */
    		return(ast_say_date_with_format_it(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "nl") ) {	/* Dutch syntax */
    		return(ast_say_date_with_format_nl(chan, time, ints, lang, format, timezone));
    	} else if (!strcasecmp(lang, "pt") ) {	/* Portuguese syntax */
    		return(ast_say_date_with_format_pt(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "tw") || !strcasecmp(lang, "zh") ) {	/* Taiwanese / Chinese syntax */
    
    		return(ast_say_date_with_format_tw(chan, time, ints, lang, format, timezone));
    
    	} else if (!strcasecmp(lang, "gr") ) {	/* Greek syntax */
    
    		return(ast_say_date_with_format_gr(chan, time, ints, lang, format, timezone));
    
    	}
    
    	/* Default to English */
    	return(ast_say_date_with_format_en(chan, time, ints, lang, format, timezone));
    }
    
    /* English syntax */
    
    int ast_say_date_with_format_en(struct ast_channel *chan, time_t time, const char *ints, const char *lang, const char *format, const char *timezone)
    
    {
    	struct tm tm;
    	int res=0, offset, sndoffset;
    	char sndfile[256], nextmsg[256];
    
    	ast_localtime(&time,&tm,timezone);
    
    
    	for (offset=0 ; format[offset] != '\0' ; offset++) {
    		ast_log(LOG_DEBUG, "Parsing %c (offset %d) in %s\n", format[offset], offset, format);
    		switch (format[offset]) {
    			/* NOTE:  if you add more options here, please try to be consistent with strftime(3) */
    			case '\'':
    				/* Literal name of a sound file */
    				sndoffset=0;
    				for (sndoffset=0 ; (format[++offset] != '\'') && (sndoffset < 256) ; sndoffset++)