1 /* 2 * Copyright (C) 2019 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License 15 */ 16 17 package android.net.netlink; 18 19 import androidx.annotation.Nullable; 20 21 import java.net.InetAddress; 22 import java.net.UnknownHostException; 23 import java.nio.ByteBuffer; 24 import java.nio.ByteOrder; 25 26 27 /** 28 * struct nlattr 29 * 30 * see: <linux_src>/include/uapi/linux/netlink.h 31 * 32 * @hide 33 */ 34 public class StructNlAttr { 35 // Already aligned. 36 public static final int NLA_HEADERLEN = 4; 37 public static final int NLA_F_NESTED = (1 << 15); 38 39 public static short makeNestedType(short type) { 40 return (short) (type | NLA_F_NESTED); 41 } 42 43 // Return a (length, type) object only, without consuming any bytes in 44 // |byteBuffer| and without copying or interpreting any value bytes. 45 // This is used for scanning over a packed set of struct nlattr's, 46 // looking for instances of a particular type. 47 public static StructNlAttr peek(ByteBuffer byteBuffer) { 48 if (byteBuffer == null || byteBuffer.remaining() < NLA_HEADERLEN) { 49 return null; 50 } 51 final int baseOffset = byteBuffer.position(); 52 53 final StructNlAttr struct = new StructNlAttr(); 54 final ByteOrder originalOrder = byteBuffer.order(); 55 byteBuffer.order(ByteOrder.nativeOrder()); 56 try { 57 struct.nla_len = byteBuffer.getShort(); 58 struct.nla_type = byteBuffer.getShort(); 59 } finally { 60 byteBuffer.order(originalOrder); 61 } 62 63 byteBuffer.position(baseOffset); 64 if (struct.nla_len < NLA_HEADERLEN) { 65 // Malformed. 66 return null; 67 } 68 return struct; 69 } 70 71 public static StructNlAttr parse(ByteBuffer byteBuffer) { 72 final StructNlAttr struct = peek(byteBuffer); 73 if (struct == null || byteBuffer.remaining() < struct.getAlignedLength()) { 74 return null; 75 } 76 77 final int baseOffset = byteBuffer.position(); 78 byteBuffer.position(baseOffset + NLA_HEADERLEN); 79 80 int valueLen = ((int) struct.nla_len) & 0xffff; 81 valueLen -= NLA_HEADERLEN; 82 if (valueLen > 0) { 83 struct.nla_value = new byte[valueLen]; 84 byteBuffer.get(struct.nla_value, 0, valueLen); 85 byteBuffer.position(baseOffset + struct.getAlignedLength()); 86 } 87 return struct; 88 } 89 90 /** 91 * Find next netlink attribute with a given type from {@link ByteBuffer}. 92 * 93 * @param attrType The given netlink attribute type is requested for. 94 * @param byteBuffer The buffer from which to find the netlink attribute. 95 * @return the found netlink attribute, or {@code null} if the netlink attribute could not be 96 * found or parsed successfully (for example, if it was truncated). 97 */ 98 @Nullable 99 public static StructNlAttr findNextAttrOfType(short attrType, 100 @Nullable ByteBuffer byteBuffer) { 101 while (byteBuffer != null && byteBuffer.remaining() > 0) { 102 final StructNlAttr nlAttr = StructNlAttr.peek(byteBuffer); 103 if (nlAttr == null) { 104 break; 105 } 106 if (nlAttr.nla_type == attrType) { 107 return StructNlAttr.parse(byteBuffer); 108 } 109 if (byteBuffer.remaining() < nlAttr.getAlignedLength()) { 110 break; 111 } 112 byteBuffer.position(byteBuffer.position() + nlAttr.getAlignedLength()); 113 } 114 return null; 115 } 116 117 public short nla_len = (short) NLA_HEADERLEN; 118 public short nla_type; 119 public byte[] nla_value; 120 121 public StructNlAttr() {} 122 123 public StructNlAttr(short type, byte value) { 124 nla_type = type; 125 setValue(new byte[1]); 126 nla_value[0] = value; 127 } 128 129 public StructNlAttr(short type, short value) { 130 this(type, value, ByteOrder.nativeOrder()); 131 } 132 133 public StructNlAttr(short type, short value, ByteOrder order) { 134 nla_type = type; 135 setValue(new byte[Short.BYTES]); 136 final ByteBuffer buf = getValueAsByteBuffer(); 137 final ByteOrder originalOrder = buf.order(); 138 try { 139 buf.order(order); 140 buf.putShort(value); 141 } finally { 142 buf.order(originalOrder); 143 } 144 } 145 146 public StructNlAttr(short type, int value) { 147 this(type, value, ByteOrder.nativeOrder()); 148 } 149 150 public StructNlAttr(short type, int value, ByteOrder order) { 151 nla_type = type; 152 setValue(new byte[Integer.BYTES]); 153 final ByteBuffer buf = getValueAsByteBuffer(); 154 final ByteOrder originalOrder = buf.order(); 155 try { 156 buf.order(order); 157 buf.putInt(value); 158 } finally { 159 buf.order(originalOrder); 160 } 161 } 162 163 public StructNlAttr(short type, InetAddress ip) { 164 nla_type = type; 165 setValue(ip.getAddress()); 166 } 167 168 public StructNlAttr(short type, StructNlAttr... nested) { 169 this(); 170 nla_type = makeNestedType(type); 171 172 int payloadLength = 0; 173 for (StructNlAttr nla : nested) payloadLength += nla.getAlignedLength(); 174 setValue(new byte[payloadLength]); 175 176 final ByteBuffer buf = getValueAsByteBuffer(); 177 for (StructNlAttr nla : nested) { 178 nla.pack(buf); 179 } 180 } 181 182 public int getAlignedLength() { 183 return NetlinkConstants.alignedLengthOf(nla_len); 184 } 185 186 /** 187 * Get attribute value as BE16. 188 */ 189 public short getValueAsBe16(short defaultValue) { 190 final ByteBuffer byteBuffer = getValueAsByteBuffer(); 191 if (byteBuffer == null || byteBuffer.remaining() != Short.BYTES) { 192 return defaultValue; 193 } 194 final ByteOrder originalOrder = byteBuffer.order(); 195 try { 196 byteBuffer.order(ByteOrder.BIG_ENDIAN); 197 return byteBuffer.getShort(); 198 } finally { 199 byteBuffer.order(originalOrder); 200 } 201 } 202 203 public int getValueAsBe32(int defaultValue) { 204 final ByteBuffer byteBuffer = getValueAsByteBuffer(); 205 if (byteBuffer == null || byteBuffer.remaining() != Integer.BYTES) { 206 return defaultValue; 207 } 208 final ByteOrder originalOrder = byteBuffer.order(); 209 try { 210 byteBuffer.order(ByteOrder.BIG_ENDIAN); 211 return byteBuffer.getInt(); 212 } finally { 213 byteBuffer.order(originalOrder); 214 } 215 } 216 217 public ByteBuffer getValueAsByteBuffer() { 218 if (nla_value == null) { return null; } 219 final ByteBuffer byteBuffer = ByteBuffer.wrap(nla_value); 220 // By convention, all buffers in this library are in native byte order because netlink is in 221 // native byte order. It's the order that is used by NetlinkSocket.recvMessage and the only 222 // order accepted by NetlinkMessage.parse. 223 byteBuffer.order(ByteOrder.nativeOrder()); 224 return byteBuffer; 225 } 226 227 /** 228 * Get attribute value as byte. 229 */ 230 public byte getValueAsByte(byte defaultValue) { 231 final ByteBuffer byteBuffer = getValueAsByteBuffer(); 232 if (byteBuffer == null || byteBuffer.remaining() != Byte.BYTES) { 233 return defaultValue; 234 } 235 return getValueAsByteBuffer().get(); 236 } 237 238 public int getValueAsInt(int defaultValue) { 239 final ByteBuffer byteBuffer = getValueAsByteBuffer(); 240 if (byteBuffer == null || byteBuffer.remaining() != Integer.BYTES) { 241 return defaultValue; 242 } 243 return getValueAsByteBuffer().getInt(); 244 } 245 246 public InetAddress getValueAsInetAddress() { 247 if (nla_value == null) { return null; } 248 249 try { 250 return InetAddress.getByAddress(nla_value); 251 } catch (UnknownHostException ignored) { 252 return null; 253 } 254 } 255 256 public void pack(ByteBuffer byteBuffer) { 257 final ByteOrder originalOrder = byteBuffer.order(); 258 final int originalPosition = byteBuffer.position(); 259 260 byteBuffer.order(ByteOrder.nativeOrder()); 261 try { 262 byteBuffer.putShort(nla_len); 263 byteBuffer.putShort(nla_type); 264 if (nla_value != null) byteBuffer.put(nla_value); 265 } finally { 266 byteBuffer.order(originalOrder); 267 } 268 byteBuffer.position(originalPosition + getAlignedLength()); 269 } 270 271 private void setValue(byte[] value) { 272 nla_value = value; 273 nla_len = (short) (NLA_HEADERLEN + ((nla_value != null) ? nla_value.length : 0)); 274 } 275 276 @Override 277 public String toString() { 278 return "StructNlAttr{ " 279 + "nla_len{" + nla_len + "}, " 280 + "nla_type{" + nla_type + "}, " 281 + "nla_value{" + NetlinkConstants.hexify(nla_value) + "}, " 282 + "}"; 283 } 284 } 285