marytts.unitselection.data.FeatureFileReader Maven / Gradle / Ivy
The newest version!
/**
* Copyright 2006 DFKI GmbH.
* All Rights Reserved. Use is subject to license terms.
*
* This file is part of MARY TTS.
*
* MARY TTS is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, version 3 of the License.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program. If not, see .
*
*/
package marytts.unitselection.data;
import java.io.BufferedInputStream;
import java.io.DataInputStream;
import java.io.FileInputStream;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import marytts.exceptions.MaryConfigurationException;
import marytts.features.FeatureDefinition;
import marytts.features.FeatureVector;
import marytts.util.data.MaryHeader;
public class FeatureFileReader {
protected MaryHeader hdr;
protected FeatureDefinition featureDefinition;
protected FeatureVector[] featureVectors;
/**
* Get a feature file reader representing the given feature file.
*
* @param fileName
* the filename of a valid feature file.
* @return a feature file object representing the given file.
* @throws IOException
* if there was a problem reading the file
* @throws MaryConfigurationException
* if the file is not a valid feature file.
*/
public static FeatureFileReader getFeatureFileReader(String fileName) throws IOException, MaryConfigurationException {
int fileType = MaryHeader.peekFileType(fileName);
if (fileType == MaryHeader.UNITFEATS)
return new FeatureFileReader(fileName);
else if (fileType == MaryHeader.HALFPHONE_UNITFEATS)
return new HalfPhoneFeatureFileReader(fileName);
throw new MaryConfigurationException("File " + fileName + ": Type " + fileType + " is not a known unit feature file type");
}
/**
* Empty constructor; need to call load() separately when using this.
*
* @see #load(String fileName)
*/
public FeatureFileReader() {
}
public FeatureFileReader(String fileName) throws IOException, MaryConfigurationException {
load(fileName);
}
public void load(String fileName) throws IOException, MaryConfigurationException {
loadFromByteBuffer(fileName);
}
protected void loadFromStream(String fileName) throws IOException, MaryConfigurationException {
/* Open the file */
DataInputStream dis = null;
dis = new DataInputStream(new BufferedInputStream(new FileInputStream(fileName)));
/* Load the Mary header */
hdr = new MaryHeader(dis);
if (hdr.getType() != MaryHeader.UNITFEATS && hdr.getType() != MaryHeader.HALFPHONE_UNITFEATS) {
throw new IOException("File [" + fileName + "] is not a valid Mary feature file.");
}
featureDefinition = new FeatureDefinition(dis);
int numberOfUnits = dis.readInt();
featureVectors = new FeatureVector[numberOfUnits];
for (int i = 0; i < numberOfUnits; i++) {
featureVectors[i] = featureDefinition.readFeatureVector(i, dis);
}
}
protected void loadFromByteBuffer(String fileName) throws IOException, MaryConfigurationException {
/* Open the file */
FileInputStream fis = new FileInputStream(fileName);
FileChannel fc = fis.getChannel();
ByteBuffer bb = fc.map(FileChannel.MapMode.READ_ONLY, 0, fc.size());
fis.close();
/* Load the Mary header */
hdr = new MaryHeader(bb);
if (hdr.getType() != MaryHeader.UNITFEATS && hdr.getType() != MaryHeader.HALFPHONE_UNITFEATS) {
throw new MaryConfigurationException("File [" + fileName + "] is not a valid Mary feature file.");
}
featureDefinition = new FeatureDefinition(bb);
int numberOfUnits = bb.getInt();
featureVectors = new FeatureVector[numberOfUnits];
for (int i = 0; i < numberOfUnits; i++) {
featureVectors[i] = featureDefinition.readFeatureVector(i, bb);
}
}
/**
* Get the unit feature vector for the given unit index number.
*
* @param unitIndex
* the absolute index number of a unit in the database
* @return the corresponding feature vector
*/
public FeatureVector getFeatureVector(int unitIndex) {
return featureVectors[unitIndex];
}
/**
* Return a shallow copy of the array of feature vectors.
*
* @return a new array containing the internal feature vectors
*/
public FeatureVector[] getCopyOfFeatureVectors() {
return (FeatureVector[]) featureVectors.clone();
}
/**
* Return the internal array of feature vectors.
*
* @return the internal array of feature vectors.
*/
public FeatureVector[] getFeatureVectors() {
return featureVectors;
}
/**
* feature vector mapping according to new feature definition Note: The new feature definition should be a subset of original
* feature definition
*
* @param newFeatureDefinition
* newFeatureDefinition
* @return newFV
*/
public FeatureVector[] featureVectorMapping(FeatureDefinition newFeatureDefinition) {
if (!this.featureDefinition.contains(newFeatureDefinition)) {
throw new RuntimeException("the new feature definition is not a subset of original feature definition");
}
int numberOfFeatures = newFeatureDefinition.getNumberOfFeatures();
int noByteFeatures = newFeatureDefinition.getNumberOfByteFeatures();
int noShortFeatures = newFeatureDefinition.getNumberOfShortFeatures();
int noContiniousFeatures = newFeatureDefinition.getNumberOfContinuousFeatures();
if (numberOfFeatures != (noByteFeatures + noShortFeatures + noContiniousFeatures)) {
throw new RuntimeException("The sum of byte, short and continious features are not equal to number of features");
}
String[] featureNames = new String[numberOfFeatures];
for (int j = 0; j < numberOfFeatures; j++) {
featureNames[j] = newFeatureDefinition.getFeatureName(j);
}
int[] featureIndexes = featureDefinition.getFeatureIndexArray(featureNames);
FeatureVector[] newFV = new FeatureVector[this.getNumberOfUnits()];
for (int i = 0; i < this.getNumberOfUnits(); i++) {
// create features array
byte[] byteFeatures = new byte[noByteFeatures];
short[] shortFeatures = new short[noShortFeatures];
float[] continiousFeatures = new float[noContiniousFeatures];
int countByteFeatures = 0;
int countShortFeatures = 0;
int countFloatFeatures = 0;
for (int j = 0; j < featureIndexes.length; j++) {
if (newFeatureDefinition.isByteFeature(j)) {
byteFeatures[countByteFeatures++] = featureVectors[i].getByteFeature(featureIndexes[j]);
} else if (newFeatureDefinition.isShortFeature(j)) {
shortFeatures[countShortFeatures++] = featureVectors[i].getShortFeature(featureIndexes[j]);
} else if (newFeatureDefinition.isContinuousFeature(j)) {
continiousFeatures[countFloatFeatures++] = featureVectors[i].getContinuousFeature(featureIndexes[j]);
}
}
newFV[i] = newFeatureDefinition.toFeatureVector(i, byteFeatures, shortFeatures, continiousFeatures);
}
return newFV;
}
/**
* Get the unit feature vector for the given unit.
*
* @param unit
* a unit in the database
* @return the corresponding feature vector
*/
public FeatureVector getFeatureVector(Unit unit) {
return featureVectors[unit.index];
}
public FeatureDefinition getFeatureDefinition() {
return featureDefinition;
}
public int getNumberOfUnits() {
return (featureVectors.length);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy